2013-07-17 19:30:41 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
2014-01-27 19:52:51 +00:00
|
|
|
|
2014-06-19 19:32:29 +00:00
|
|
|
#include "Benchmark.h"
|
2013-07-17 19:30:41 +00:00
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkPaint.h"
|
|
|
|
#include "SkRandom.h"
|
|
|
|
#include "SkShader.h"
|
|
|
|
#include "SkString.h"
|
|
|
|
|
2014-02-19 19:30:42 +00:00
|
|
|
#if SK_SUPPORT_GPU
|
|
|
|
#include "GrDrawTargetCaps.h"
|
2014-06-19 19:32:29 +00:00
|
|
|
#include "GrTest.h"
|
2014-02-19 19:30:42 +00:00
|
|
|
#endif
|
|
|
|
|
2013-07-17 19:30:41 +00:00
|
|
|
enum Flags {
|
|
|
|
kBig_Flag = 1 << 0,
|
|
|
|
kAA_Flag = 1 << 1
|
|
|
|
};
|
|
|
|
|
|
|
|
#define FLAGS00 Flags(0)
|
|
|
|
#define FLAGS01 Flags(kBig_Flag)
|
|
|
|
#define FLAGS10 Flags(kAA_Flag)
|
|
|
|
#define FLAGS11 Flags(kBig_Flag | kAA_Flag)
|
|
|
|
|
|
|
|
static const int points[] = {
|
|
|
|
10, 10, 15, 5, 20, 20,
|
|
|
|
30, 5, 25, 20, 15, 12,
|
|
|
|
21, 21, 30, 30, 12, 4,
|
|
|
|
32, 28, 20, 18, 12, 10
|
|
|
|
};
|
|
|
|
|
|
|
|
static const int kMaxPathSize = 10;
|
|
|
|
|
2014-06-19 19:32:29 +00:00
|
|
|
class HairlinePathBench : public Benchmark {
|
2013-07-17 19:30:41 +00:00
|
|
|
public:
|
2013-09-13 19:52:27 +00:00
|
|
|
HairlinePathBench(Flags flags) : fFlags(flags) {
|
2013-07-17 19:30:41 +00:00
|
|
|
fPaint.setStyle(SkPaint::kStroke_Style);
|
|
|
|
fPaint.setStrokeWidth(SkIntToScalar(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void appendName(SkString*) = 0;
|
|
|
|
virtual void makePath(SkPath*) = 0;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual const char* onGetName() SK_OVERRIDE {
|
|
|
|
fName.printf("path_hairline_%s_%s_",
|
|
|
|
fFlags & kBig_Flag ? "big" : "small",
|
|
|
|
fFlags & kAA_Flag ? "AA" : "noAA");
|
|
|
|
this->appendName(&fName);
|
|
|
|
return fName.c_str();
|
|
|
|
}
|
|
|
|
|
2013-12-03 18:17:16 +00:00
|
|
|
virtual void onDraw(const int loops, SkCanvas* canvas) SK_OVERRIDE {
|
2013-07-17 19:30:41 +00:00
|
|
|
SkPaint paint(fPaint);
|
|
|
|
this->setupPaint(&paint);
|
|
|
|
|
|
|
|
paint.setAntiAlias(fFlags & kAA_Flag ? true : false);
|
2013-07-18 07:00:56 +00:00
|
|
|
|
2013-07-17 19:30:41 +00:00
|
|
|
SkPath path;
|
|
|
|
this->makePath(&path);
|
|
|
|
if (fFlags & kBig_Flag) {
|
|
|
|
SkMatrix m;
|
|
|
|
m.setScale(SkIntToScalar(3), SkIntToScalar(3));
|
|
|
|
path.transform(m);
|
|
|
|
}
|
|
|
|
|
2013-12-03 18:17:16 +00:00
|
|
|
for (int i = 0; i < loops; i++) {
|
2013-07-17 19:30:41 +00:00
|
|
|
canvas->drawPath(path, paint);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SkPaint fPaint;
|
|
|
|
SkString fName;
|
|
|
|
Flags fFlags;
|
2014-06-19 19:32:29 +00:00
|
|
|
typedef Benchmark INHERITED;
|
2013-07-17 19:30:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class LinePathBench : public HairlinePathBench {
|
|
|
|
public:
|
2013-09-13 19:52:27 +00:00
|
|
|
LinePathBench(Flags flags) : INHERITED(flags) {}
|
2013-07-17 19:30:41 +00:00
|
|
|
|
|
|
|
virtual void appendName(SkString* name) SK_OVERRIDE {
|
|
|
|
name->append("line");
|
|
|
|
}
|
|
|
|
virtual void makePath(SkPath* path) SK_OVERRIDE {
|
2013-09-09 20:09:12 +00:00
|
|
|
SkRandom rand;
|
2013-07-17 19:30:41 +00:00
|
|
|
int size = SK_ARRAY_COUNT(points);
|
|
|
|
int hSize = size / 2;
|
|
|
|
for (int i = 0; i < kMaxPathSize; ++i) {
|
|
|
|
int xTrans = 10 + 40 * (i%(kMaxPathSize/2));
|
|
|
|
int yTrans = 0;
|
|
|
|
if (i > kMaxPathSize/2 - 1) {
|
|
|
|
yTrans = 40;
|
|
|
|
}
|
|
|
|
int base1 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base2 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base3 = 2 * rand.nextULessThan(hSize);
|
|
|
|
path->moveTo(SkIntToScalar(points[base1] + xTrans),
|
|
|
|
SkIntToScalar(points[base1+1] + yTrans));
|
|
|
|
path->lineTo(SkIntToScalar(points[base2] + xTrans),
|
|
|
|
SkIntToScalar(points[base2+1] + yTrans));
|
|
|
|
path->lineTo(SkIntToScalar(points[base3] + xTrans),
|
|
|
|
SkIntToScalar(points[base3+1] + yTrans));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
typedef HairlinePathBench INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
class QuadPathBench : public HairlinePathBench {
|
|
|
|
public:
|
2013-09-13 19:52:27 +00:00
|
|
|
QuadPathBench(Flags flags) : INHERITED(flags) {}
|
2013-07-17 19:30:41 +00:00
|
|
|
|
|
|
|
virtual void appendName(SkString* name) SK_OVERRIDE {
|
|
|
|
name->append("quad");
|
|
|
|
}
|
|
|
|
virtual void makePath(SkPath* path) SK_OVERRIDE {
|
2013-09-09 20:09:12 +00:00
|
|
|
SkRandom rand;
|
2013-07-17 19:30:41 +00:00
|
|
|
int size = SK_ARRAY_COUNT(points);
|
|
|
|
int hSize = size / 2;
|
|
|
|
for (int i = 0; i < kMaxPathSize; ++i) {
|
|
|
|
int xTrans = 10 + 40 * (i%(kMaxPathSize/2));
|
|
|
|
int yTrans = 0;
|
|
|
|
if (i > kMaxPathSize/2 - 1) {
|
|
|
|
yTrans = 40;
|
|
|
|
}
|
|
|
|
int base1 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base2 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base3 = 2 * rand.nextULessThan(hSize);
|
|
|
|
path->moveTo(SkIntToScalar(points[base1] + xTrans),
|
|
|
|
SkIntToScalar(points[base1+1] + yTrans));
|
|
|
|
path->quadTo(SkIntToScalar(points[base2] + xTrans),
|
|
|
|
SkIntToScalar(points[base2+1] + yTrans),
|
|
|
|
SkIntToScalar(points[base3] + xTrans),
|
|
|
|
SkIntToScalar(points[base3+1] + yTrans));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
typedef HairlinePathBench INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ConicPathBench : public HairlinePathBench {
|
|
|
|
public:
|
2013-09-13 19:52:27 +00:00
|
|
|
ConicPathBench(Flags flags) : INHERITED(flags) {}
|
2013-07-17 19:30:41 +00:00
|
|
|
|
|
|
|
virtual void appendName(SkString* name) SK_OVERRIDE {
|
|
|
|
name->append("conic");
|
|
|
|
}
|
|
|
|
virtual void makePath(SkPath* path) SK_OVERRIDE {
|
2013-09-09 20:09:12 +00:00
|
|
|
SkRandom rand;
|
|
|
|
SkRandom randWeight;
|
2013-07-17 19:30:41 +00:00
|
|
|
int size = SK_ARRAY_COUNT(points);
|
|
|
|
int hSize = size / 2;
|
|
|
|
for (int i = 0; i < kMaxPathSize; ++i) {
|
|
|
|
int xTrans = 10 + 40 * (i%(kMaxPathSize/2));
|
|
|
|
int yTrans = 0;
|
|
|
|
if (i > kMaxPathSize/2 - 1) {
|
|
|
|
yTrans = 40;
|
|
|
|
}
|
|
|
|
int base1 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base2 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base3 = 2 * rand.nextULessThan(hSize);
|
|
|
|
float weight = randWeight.nextRangeF(0.0f, 2.0f);
|
|
|
|
path->moveTo(SkIntToScalar(points[base1] + xTrans),
|
|
|
|
SkIntToScalar(points[base1+1] + yTrans));
|
|
|
|
path->conicTo(SkIntToScalar(points[base2] + xTrans),
|
|
|
|
SkIntToScalar(points[base2+1] + yTrans),
|
|
|
|
SkIntToScalar(points[base3] + xTrans),
|
|
|
|
SkIntToScalar(points[base3+1] + yTrans),
|
|
|
|
weight);
|
|
|
|
}
|
|
|
|
}
|
2014-01-27 19:52:51 +00:00
|
|
|
|
|
|
|
virtual void onDraw(const int loops, SkCanvas* canvas) SK_OVERRIDE {
|
|
|
|
#if SK_SUPPORT_GPU
|
|
|
|
GrContext* context = canvas->getGrContext();
|
|
|
|
// This is a workaround for skbug.com/2078. See also skbug.com/2033.
|
|
|
|
if (NULL != context) {
|
|
|
|
GrTestTarget tt;
|
|
|
|
context->getTestTarget(&tt);
|
|
|
|
if (tt.target()->caps()->pathRenderingSupport()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
INHERITED::onDraw(loops, canvas);
|
|
|
|
}
|
|
|
|
|
2013-07-17 19:30:41 +00:00
|
|
|
private:
|
|
|
|
typedef HairlinePathBench INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CubicPathBench : public HairlinePathBench {
|
|
|
|
public:
|
2013-09-13 19:52:27 +00:00
|
|
|
CubicPathBench(Flags flags) : INHERITED(flags) {}
|
2013-07-17 19:30:41 +00:00
|
|
|
|
|
|
|
virtual void appendName(SkString* name) SK_OVERRIDE {
|
|
|
|
name->append("cubic");
|
|
|
|
}
|
|
|
|
virtual void makePath(SkPath* path) SK_OVERRIDE {
|
2013-09-09 20:09:12 +00:00
|
|
|
SkRandom rand;
|
2013-07-17 19:30:41 +00:00
|
|
|
int size = SK_ARRAY_COUNT(points);
|
|
|
|
int hSize = size / 2;
|
|
|
|
for (int i = 0; i < kMaxPathSize; ++i) {
|
|
|
|
int xTrans = 10 + 40 * (i%(kMaxPathSize/2));
|
|
|
|
int yTrans = 0;
|
|
|
|
if (i > kMaxPathSize/2 - 1) {
|
|
|
|
yTrans = 40;
|
|
|
|
}
|
|
|
|
int base1 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base2 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base3 = 2 * rand.nextULessThan(hSize);
|
|
|
|
int base4 = 2 * rand.nextULessThan(hSize);
|
|
|
|
path->moveTo(SkIntToScalar(points[base1] + xTrans),
|
|
|
|
SkIntToScalar(points[base1+1] + yTrans));
|
|
|
|
path->cubicTo(SkIntToScalar(points[base2] + xTrans),
|
|
|
|
SkIntToScalar(points[base2+1] + yTrans),
|
|
|
|
SkIntToScalar(points[base3] + xTrans),
|
|
|
|
SkIntToScalar(points[base3+1] + yTrans),
|
|
|
|
SkIntToScalar(points[base4] + xTrans),
|
|
|
|
SkIntToScalar(points[base4+1] + yTrans));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
typedef HairlinePathBench INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
// FLAG00 - no AA, small
|
|
|
|
// FLAG01 - no AA, small
|
|
|
|
// FLAG10 - AA, big
|
|
|
|
// FLAG11 - AA, big
|
|
|
|
|
2013-09-13 19:52:27 +00:00
|
|
|
DEF_BENCH( return new LinePathBench(FLAGS00); )
|
|
|
|
DEF_BENCH( return new LinePathBench(FLAGS01); )
|
|
|
|
DEF_BENCH( return new LinePathBench(FLAGS10); )
|
|
|
|
DEF_BENCH( return new LinePathBench(FLAGS11); )
|
2013-07-17 19:30:41 +00:00
|
|
|
|
2013-09-13 19:52:27 +00:00
|
|
|
DEF_BENCH( return new QuadPathBench(FLAGS00); )
|
|
|
|
DEF_BENCH( return new QuadPathBench(FLAGS01); )
|
|
|
|
DEF_BENCH( return new QuadPathBench(FLAGS10); )
|
|
|
|
DEF_BENCH( return new QuadPathBench(FLAGS11); )
|
2013-07-17 19:30:41 +00:00
|
|
|
|
|
|
|
// Don't have default path renderer for conics yet on GPU, so must use AA
|
2013-09-13 19:52:27 +00:00
|
|
|
// DEF_BENCH( return new ConicPathBench(FLAGS00); )
|
|
|
|
// DEF_BENCH( return new ConicPathBench(FLAGS01); )
|
|
|
|
DEF_BENCH( return new ConicPathBench(FLAGS10); )
|
|
|
|
DEF_BENCH( return new ConicPathBench(FLAGS11); )
|
|
|
|
|
|
|
|
DEF_BENCH( return new CubicPathBench(FLAGS00); )
|
|
|
|
DEF_BENCH( return new CubicPathBench(FLAGS01); )
|
|
|
|
DEF_BENCH( return new CubicPathBench(FLAGS10); )
|
|
|
|
DEF_BENCH( return new CubicPathBench(FLAGS11); )
|