e25c114f94
Change-Id: I63ce1643364c6d2dd50e3622d66c64bdb3091418 Reviewed-on: https://skia-review.googlesource.com/c/skia/+/550704 Reviewed-by: John Stiles <johnstiles@google.com> Commit-Queue: Herb Derby <herb@google.com>
485 lines
15 KiB
C++
485 lines
15 KiB
C++
/*
|
|
* 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/SkString.h"
|
|
#include "include/core/SkStrokeRec.h"
|
|
#include "include/effects/SkDashPathEffect.h"
|
|
#include "include/private/SkTDArray.h"
|
|
#include "include/utils/SkRandom.h"
|
|
|
|
|
|
/*
|
|
* Cases to consider:
|
|
*
|
|
* 1. antialiasing on/off (esp. width <= 1)
|
|
* 2. strokewidth == 0, 1, 2
|
|
* 3. hline, vline, diagonal, rect, oval
|
|
* 4. dots [1,1] ([N,N] where N=strokeWidth?) or arbitrary (e.g. [2,1] or [1,2,3,2])
|
|
*/
|
|
static void path_hline(SkPath* path) {
|
|
path->moveTo(SkIntToScalar(10), SkIntToScalar(10));
|
|
path->lineTo(SkIntToScalar(600), SkIntToScalar(10));
|
|
}
|
|
|
|
class DashBench : public Benchmark {
|
|
protected:
|
|
SkString fName;
|
|
SkTDArray<SkScalar> fIntervals;
|
|
int fWidth;
|
|
SkPoint fPts[2];
|
|
bool fDoClip;
|
|
|
|
public:
|
|
DashBench(const SkScalar intervals[], int count, int width,
|
|
bool doClip = false) {
|
|
fIntervals.append(count, intervals);
|
|
for (int i = 0; i < count; ++i) {
|
|
fIntervals[i] *= width;
|
|
}
|
|
fWidth = width;
|
|
fName.printf("dash_%d_%s", width, doClip ? "clipped" : "noclip");
|
|
fDoClip = doClip;
|
|
|
|
fPts[0].set(SkIntToScalar(10), SkIntToScalar(10));
|
|
fPts[1].set(SkIntToScalar(600), SkIntToScalar(10));
|
|
}
|
|
|
|
virtual void makePath(SkPath* path) {
|
|
path_hline(path);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkPaint paint;
|
|
this->setupPaint(&paint);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
paint.setStrokeWidth(SkIntToScalar(fWidth));
|
|
paint.setAntiAlias(false);
|
|
|
|
SkPath path;
|
|
this->makePath(&path);
|
|
|
|
paint.setPathEffect(SkDashPathEffect::Make(fIntervals.begin(), fIntervals.count(), 0));
|
|
|
|
if (fDoClip) {
|
|
SkRect r = path.getBounds();
|
|
r.inset(-SkIntToScalar(20), -SkIntToScalar(20));
|
|
// now move it so we don't intersect
|
|
r.offset(0, r.height() * 3 / 2);
|
|
canvas->clipRect(r);
|
|
}
|
|
|
|
this->handlePath(canvas, path, paint, loops);
|
|
}
|
|
|
|
virtual void handlePath(SkCanvas* canvas, const SkPath& path,
|
|
const SkPaint& paint, int N) {
|
|
for (int i = 0; i < N; ++i) {
|
|
// canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, paint);
|
|
canvas->drawPath(path, paint);
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
class RectDashBench : public DashBench {
|
|
public:
|
|
RectDashBench(const SkScalar intervals[], int count, int width)
|
|
: INHERITED(intervals, count, width) {
|
|
fName.append("_rect");
|
|
}
|
|
|
|
protected:
|
|
void handlePath(SkCanvas* canvas, const SkPath& path, const SkPaint& paint, int N) override {
|
|
SkPoint pts[2];
|
|
if (!path.isLine(pts) || pts[0].fY != pts[1].fY) {
|
|
this->INHERITED::handlePath(canvas, path, paint, N);
|
|
} else {
|
|
SkRect rect;
|
|
rect.fLeft = pts[0].fX;
|
|
rect.fTop = pts[0].fY - paint.getStrokeWidth() / 2;
|
|
rect.fRight = rect.fLeft + SkIntToScalar(fWidth);
|
|
rect.fBottom = rect.fTop + paint.getStrokeWidth();
|
|
|
|
SkPaint p(paint);
|
|
p.setStyle(SkPaint::kFill_Style);
|
|
p.setPathEffect(nullptr);
|
|
|
|
int count = SkScalarRoundToInt((pts[1].fX - pts[0].fX) / (2*fWidth));
|
|
SkScalar dx = SkIntToScalar(2 * fWidth);
|
|
|
|
for (int i = 0; i < N*10; ++i) {
|
|
SkRect r = rect;
|
|
for (int j = 0; j < count; ++j) {
|
|
canvas->drawRect(r, p);
|
|
r.offset(dx, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = DashBench;
|
|
};
|
|
|
|
static void make_unit_star(SkPath* path, int n) {
|
|
SkScalar rad = -SK_ScalarPI / 2;
|
|
const SkScalar drad = (n >> 1) * SK_ScalarPI * 2 / n;
|
|
|
|
path->moveTo(0, -SK_Scalar1);
|
|
for (int i = 1; i < n; i++) {
|
|
rad += drad;
|
|
path->lineTo(SkScalarCos(rad), SkScalarSin(rad));
|
|
}
|
|
path->close();
|
|
}
|
|
|
|
static void make_poly(SkPath* path) {
|
|
make_unit_star(path, 9);
|
|
const SkMatrix matrix = SkMatrix::Scale(100, 100);
|
|
path->transform(matrix);
|
|
}
|
|
|
|
static void make_quad(SkPath* path) {
|
|
SkScalar x0 = SkIntToScalar(10);
|
|
SkScalar y0 = SkIntToScalar(10);
|
|
path->moveTo(x0, y0);
|
|
path->quadTo(x0, y0 + 400 * SK_Scalar1,
|
|
x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1);
|
|
}
|
|
|
|
static void make_cubic(SkPath* path) {
|
|
SkScalar x0 = SkIntToScalar(10);
|
|
SkScalar y0 = SkIntToScalar(10);
|
|
path->moveTo(x0, y0);
|
|
path->cubicTo(x0, y0 + 400 * SK_Scalar1,
|
|
x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1,
|
|
x0 + 600 * SK_Scalar1, y0);
|
|
}
|
|
|
|
class MakeDashBench : public Benchmark {
|
|
SkString fName;
|
|
SkPath fPath;
|
|
sk_sp<SkPathEffect> fPE;
|
|
|
|
public:
|
|
MakeDashBench(void (*proc)(SkPath*), const char name[]) {
|
|
fName.printf("makedash_%s", name);
|
|
proc(&fPath);
|
|
|
|
SkScalar vals[] = { SkIntToScalar(4), SkIntToScalar(4) };
|
|
fPE = SkDashPathEffect::Make(vals, 2, 0);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas*) override {
|
|
SkPath dst;
|
|
for (int i = 0; i < loops; ++i) {
|
|
SkStrokeRec rec(SkStrokeRec::kHairline_InitStyle);
|
|
|
|
fPE->filterPath(&dst, fPath, &rec, nullptr);
|
|
dst.rewind();
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
/*
|
|
* We try to special case square dashes (intervals are equal to strokewidth).
|
|
*/
|
|
class DashLineBench : public Benchmark {
|
|
SkString fName;
|
|
SkScalar fStrokeWidth;
|
|
bool fIsRound;
|
|
sk_sp<SkPathEffect> fPE;
|
|
|
|
public:
|
|
DashLineBench(SkScalar width, bool isRound) {
|
|
fName.printf("dashline_%g_%s", SkScalarToFloat(width), isRound ? "circle" : "square");
|
|
fStrokeWidth = width;
|
|
fIsRound = isRound;
|
|
|
|
SkScalar vals[] = { SK_Scalar1, SK_Scalar1 };
|
|
fPE = SkDashPathEffect::Make(vals, 2, 0);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkPaint paint;
|
|
this->setupPaint(&paint);
|
|
paint.setStrokeWidth(fStrokeWidth);
|
|
paint.setStrokeCap(fIsRound ? SkPaint::kRound_Cap : SkPaint::kSquare_Cap);
|
|
paint.setPathEffect(fPE);
|
|
for (int i = 0; i < loops; ++i) {
|
|
canvas->drawLine(10 * SK_Scalar1, 10 * SK_Scalar1,
|
|
640 * SK_Scalar1, 10 * SK_Scalar1, paint);
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
class DrawPointsDashingBench : public Benchmark {
|
|
SkString fName;
|
|
int fStrokeWidth;
|
|
bool fDoAA;
|
|
|
|
sk_sp<SkPathEffect> fPathEffect;
|
|
|
|
public:
|
|
DrawPointsDashingBench(int dashLength, int strokeWidth, bool doAA)
|
|
{
|
|
fName.printf("drawpointsdash_%d_%d%s", dashLength, strokeWidth, doAA ? "_aa" : "_bw");
|
|
fStrokeWidth = strokeWidth;
|
|
fDoAA = doAA;
|
|
|
|
SkScalar vals[] = { SkIntToScalar(dashLength), SkIntToScalar(dashLength) };
|
|
fPathEffect = SkDashPathEffect::Make(vals, 2, SK_Scalar1);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkPaint p;
|
|
this->setupPaint(&p);
|
|
p.setColor(SK_ColorBLACK);
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
p.setStrokeWidth(SkIntToScalar(fStrokeWidth));
|
|
p.setPathEffect(fPathEffect);
|
|
p.setAntiAlias(fDoAA);
|
|
|
|
SkPoint pts[2] = {
|
|
{ SkIntToScalar(10), 0 },
|
|
{ SkIntToScalar(640), 0 }
|
|
};
|
|
|
|
for (int i = 0; i < loops; ++i) {
|
|
pts[0].fY = pts[1].fY = SkIntToScalar(i % 480);
|
|
canvas->drawPoints(SkCanvas::kLines_PointMode, 2, pts, p);
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
// Want to test how we handle dashing when 99% of the dash is clipped out
|
|
class GiantDashBench : public Benchmark {
|
|
SkString fName;
|
|
SkScalar fStrokeWidth;
|
|
SkPoint fPts[2];
|
|
sk_sp<SkPathEffect> fPathEffect;
|
|
|
|
public:
|
|
enum LineType {
|
|
kHori_LineType,
|
|
kVert_LineType,
|
|
kDiag_LineType,
|
|
kLineTypeCount
|
|
};
|
|
|
|
static const char* LineTypeName(LineType lt) {
|
|
static const char* gNames[] = { "hori", "vert", "diag" };
|
|
static_assert(kLineTypeCount == std::size(gNames), "names_wrong_size");
|
|
return gNames[lt];
|
|
}
|
|
|
|
GiantDashBench(LineType lt, SkScalar width) {
|
|
fName.printf("giantdashline_%s_%g", LineTypeName(lt), width);
|
|
fStrokeWidth = width;
|
|
|
|
// deliberately pick intervals that won't be caught by asPoints(), so
|
|
// we can test the filterPath code-path.
|
|
const SkScalar intervals[] = { 20, 10, 10, 10 };
|
|
fPathEffect = SkDashPathEffect::Make(intervals, std::size(intervals), 0);
|
|
|
|
SkScalar cx = 640 / 2; // center X
|
|
SkScalar cy = 480 / 2; // center Y
|
|
SkMatrix matrix;
|
|
|
|
switch (lt) {
|
|
case kHori_LineType:
|
|
matrix.setIdentity();
|
|
break;
|
|
case kVert_LineType:
|
|
matrix.setRotate(90, cx, cy);
|
|
break;
|
|
case kDiag_LineType:
|
|
matrix.setRotate(45, cx, cy);
|
|
break;
|
|
case kLineTypeCount:
|
|
// Not a real enum value.
|
|
break;
|
|
}
|
|
|
|
const SkScalar overshoot = 100*1000;
|
|
const SkPoint pts[2] = {
|
|
{ -overshoot, cy }, { 640 + overshoot, cy }
|
|
};
|
|
matrix.mapPoints(fPts, pts, 2);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkPaint p;
|
|
this->setupPaint(&p);
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
p.setStrokeWidth(fStrokeWidth);
|
|
p.setPathEffect(fPathEffect);
|
|
|
|
for (int i = 0; i < loops; i++) {
|
|
canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, p);
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
// Want to test how we draw a dashed grid (like what is used in spreadsheets) of many
|
|
// small dashed lines switching back and forth between horizontal and vertical
|
|
class DashGridBench : public Benchmark {
|
|
SkString fName;
|
|
int fStrokeWidth;
|
|
bool fDoAA;
|
|
|
|
sk_sp<SkPathEffect> fPathEffect;
|
|
|
|
public:
|
|
DashGridBench(int dashLength, int strokeWidth, bool doAA) {
|
|
fName.printf("dashgrid_%d_%d%s", dashLength, strokeWidth, doAA ? "_aa" : "_bw");
|
|
fStrokeWidth = strokeWidth;
|
|
fDoAA = doAA;
|
|
|
|
SkScalar vals[] = { SkIntToScalar(dashLength), SkIntToScalar(dashLength) };
|
|
fPathEffect = SkDashPathEffect::Make(vals, 2, SK_Scalar1);
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
SkPaint p;
|
|
this->setupPaint(&p);
|
|
p.setColor(SK_ColorBLACK);
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
p.setStrokeWidth(SkIntToScalar(fStrokeWidth));
|
|
p.setPathEffect(fPathEffect);
|
|
p.setAntiAlias(fDoAA);
|
|
|
|
SkPoint pts[4] = {
|
|
{ SkIntToScalar(0), 20.5f },
|
|
{ SkIntToScalar(20), 20.5f },
|
|
{ 20.5f, SkIntToScalar(0) },
|
|
{ 20.5f, SkIntToScalar(20) }
|
|
};
|
|
|
|
for (int i = 0; i < loops; ++i) {
|
|
for (int j = 0; j < 10; ++j) {
|
|
for (int k = 0; k < 10; ++k) {
|
|
// Horizontal line
|
|
SkPoint horPts[2];
|
|
horPts[0].fX = pts[0].fX + k * 22.f;
|
|
horPts[0].fY = pts[0].fY + j * 22.f;
|
|
horPts[1].fX = pts[1].fX + k * 22.f;
|
|
horPts[1].fY = pts[1].fY + j * 22.f;
|
|
canvas->drawPoints(SkCanvas::kLines_PointMode, 2, horPts, p);
|
|
|
|
// Vertical line
|
|
SkPoint vertPts[2];
|
|
vertPts[0].fX = pts[2].fX + k * 22.f;
|
|
vertPts[0].fY = pts[2].fY + j * 22.f;
|
|
vertPts[1].fX = pts[3].fX + k * 22.f;
|
|
vertPts[1].fY = pts[3].fY + j * 22.f;
|
|
canvas->drawPoints(SkCanvas::kLines_PointMode, 2, vertPts, p);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
static const SkScalar gDots[] = { SK_Scalar1, SK_Scalar1 };
|
|
|
|
#define PARAM(array) array, std::size(array)
|
|
|
|
DEF_BENCH( return new DashBench(PARAM(gDots), 0); )
|
|
DEF_BENCH( return new DashBench(PARAM(gDots), 1); )
|
|
DEF_BENCH( return new DashBench(PARAM(gDots), 1, true); )
|
|
DEF_BENCH( return new DashBench(PARAM(gDots), 4); )
|
|
DEF_BENCH( return new MakeDashBench(make_poly, "poly"); )
|
|
DEF_BENCH( return new MakeDashBench(make_quad, "quad"); )
|
|
DEF_BENCH( return new MakeDashBench(make_cubic, "cubic"); )
|
|
DEF_BENCH( return new DashLineBench(0, false); )
|
|
DEF_BENCH( return new DashLineBench(SK_Scalar1, false); )
|
|
DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, false); )
|
|
DEF_BENCH( return new DashLineBench(0, true); )
|
|
DEF_BENCH( return new DashLineBench(SK_Scalar1, true); )
|
|
DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, true); )
|
|
|
|
DEF_BENCH( return new DrawPointsDashingBench(1, 1, false); )
|
|
DEF_BENCH( return new DrawPointsDashingBench(1, 1, true); )
|
|
DEF_BENCH( return new DrawPointsDashingBench(3, 1, false); )
|
|
DEF_BENCH( return new DrawPointsDashingBench(3, 1, true); )
|
|
DEF_BENCH( return new DrawPointsDashingBench(5, 5, false); )
|
|
DEF_BENCH( return new DrawPointsDashingBench(5, 5, true); )
|
|
|
|
/* Disable the GiantDashBench for Android devices until we can better control
|
|
* the memory usage. (https://code.google.com/p/skia/issues/detail?id=1430)
|
|
*/
|
|
#ifndef SK_BUILD_FOR_ANDROID
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 0); )
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 0); )
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 0); )
|
|
|
|
// pass 2 to explicitly avoid any 1-is-the-same-as-hairline special casing
|
|
|
|
// hori_2 is just too slow to enable at the moment
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 2); )
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 2); )
|
|
DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 2); )
|
|
|
|
DEF_BENCH( return new DashGridBench(1, 1, true); )
|
|
DEF_BENCH( return new DashGridBench(1, 1, false); )
|
|
DEF_BENCH( return new DashGridBench(3, 1, true); )
|
|
DEF_BENCH( return new DashGridBench(3, 1, false); )
|
|
#endif
|