| /* |
| * 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 "Benchmark.h" |
| #include "SkBitmap.h" |
| #include "SkCanvas.h" |
| #include "SkColorPriv.h" |
| #include "SkGradientShader.h" |
| #include "SkLinearGradient.h" |
| #include "SkPaint.h" |
| #include "SkShader.h" |
| #include "SkString.h" |
| |
| struct GradData { |
| int fCount; |
| const SkColor* fColors; |
| const SkScalar* fPos; |
| const char* fName; |
| }; |
| |
| static const SkColor gColors[] = { |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, |
| SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, // 10 lines, 50 colors |
| }; |
| |
| static const SkColor gShallowColors[] = { 0xFF555555, 0xFF444444 }; |
| |
| // We have several special-cases depending on the number (and spacing) of colors, so |
| // try to exercise those here. |
| static const GradData gGradData[] = { |
| { 2, gColors, nullptr, "" }, |
| { 50, gColors, nullptr, "_hicolor" }, // many color gradient |
| { 3, gColors, nullptr, "_3color" }, |
| { 2, gShallowColors, nullptr, "_shallow" }, |
| }; |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeLinear(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| const uint32_t flags = force4f ? SkLinearGradient::kForce4fContext_PrivateFlag : 0; |
| return SkGradientShader::MakeLinear(pts, data.fColors, data.fPos, |
| data.fCount, tm, flags, nullptr); |
| } |
| |
| static sk_sp<SkShader> MakeRadial(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center; |
| center.set(SkScalarAve(pts[0].fX, pts[1].fX), |
| SkScalarAve(pts[0].fY, pts[1].fY)); |
| return SkGradientShader::MakeRadial(center, center.fX * scale, data.fColors, |
| data.fPos, data.fCount, tm); |
| } |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeSweep(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center; |
| center.set(SkScalarAve(pts[0].fX, pts[1].fX), |
| SkScalarAve(pts[0].fY, pts[1].fY)); |
| return SkGradientShader::MakeSweep(center.fX, center.fY, data.fColors, data.fPos, data.fCount); |
| } |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeConical(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center0, center1; |
| center0.set(SkScalarAve(pts[0].fX, pts[1].fX), |
| SkScalarAve(pts[0].fY, pts[1].fY)); |
| center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), |
| SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); |
| return SkGradientShader::MakeTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7, |
| center0, (pts[1].fX - pts[0].fX) / 2, |
| data.fColors, data.fPos, data.fCount, tm); |
| } |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeConicalZeroRad(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center0, center1; |
| center0.set(SkScalarAve(pts[0].fX, pts[1].fX), |
| SkScalarAve(pts[0].fY, pts[1].fY)); |
| center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5), |
| SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4)); |
| return SkGradientShader::MakeTwoPointConical(center1, 0.0, |
| center0, (pts[1].fX - pts[0].fX) / 2, |
| data.fColors, data.fPos, data.fCount, tm); |
| } |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeConicalOutside(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center0, center1; |
| SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10; |
| SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3; |
| center0.set(pts[0].fX + radius0, pts[0].fY + radius0); |
| center1.set(pts[1].fX - radius1, pts[1].fY - radius1); |
| return SkGradientShader::MakeTwoPointConical(center0, radius0, |
| center1, radius1, |
| data.fColors, data.fPos, |
| data.fCount, tm); |
| } |
| |
| /// Ignores scale |
| static sk_sp<SkShader> MakeConicalOutsideZeroRad(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| SkPoint center0, center1; |
| SkScalar radius0 = (pts[1].fX - pts[0].fX) / 10; |
| SkScalar radius1 = (pts[1].fX - pts[0].fX) / 3; |
| center0.set(pts[0].fX + radius0, pts[0].fY + radius0); |
| center1.set(pts[1].fX - radius1, pts[1].fY - radius1); |
| return SkGradientShader::MakeTwoPointConical(center0, 0.0, |
| center1, radius1, |
| data.fColors, data.fPos, |
| data.fCount, tm); |
| } |
| |
| typedef sk_sp<SkShader> (*GradMaker)(const SkPoint pts[2], const GradData& data, |
| SkShader::TileMode tm, float scale, bool force4f); |
| |
| static const struct { |
| GradMaker fMaker; |
| const char* fName; |
| } gGrads[] = { |
| { MakeLinear, "linear" }, |
| { MakeRadial, "radial1" }, |
| { MakeSweep, "sweep" }, |
| { MakeConical, "conical" }, |
| { MakeConicalZeroRad, "conicalZero" }, |
| { MakeConicalOutside, "conicalOut" }, |
| { MakeConicalOutsideZeroRad, "conicalOutZero" }, |
| }; |
| |
| enum GradType { // these must match the order in gGrads |
| kLinear_GradType, |
| kRadial_GradType, |
| kSweep_GradType, |
| kConical_GradType, |
| kConicalZero_GradType, |
| kConicalOut_GradType, |
| kConicalOutZero_GradType |
| }; |
| |
| enum GeomType { |
| kRect_GeomType, |
| kOval_GeomType |
| }; |
| |
| static const char* tilemodename(SkShader::TileMode tm) { |
| switch (tm) { |
| case SkShader::kClamp_TileMode: |
| return "clamp"; |
| case SkShader::kRepeat_TileMode: |
| return "repeat"; |
| case SkShader::kMirror_TileMode: |
| return "mirror"; |
| default: |
| SkDEBUGFAIL("unknown tilemode"); |
| return "error"; |
| } |
| } |
| |
| static const char* geomtypename(GeomType gt) { |
| switch (gt) { |
| case kRect_GeomType: |
| return "rectangle"; |
| case kOval_GeomType: |
| return "oval"; |
| default: |
| SkDEBUGFAIL("unknown geometry type"); |
| return "error"; |
| } |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| class GradientBench : public Benchmark { |
| public: |
| GradientBench(GradType gradType, |
| GradData data = gGradData[0], |
| SkShader::TileMode tm = SkShader::kClamp_TileMode, |
| GeomType geomType = kRect_GeomType, |
| float scale = 1.0f, |
| bool force4f = false) |
| : fGeomType(geomType) { |
| |
| fName.printf("gradient_%s_%s", gGrads[gradType].fName, |
| tilemodename(tm)); |
| if (geomType != kRect_GeomType) { |
| fName.appendf("_%s", geomtypename(geomType)); |
| } |
| |
| if (scale != 1.f) { |
| fName.appendf("_scale_%g", scale); |
| } |
| |
| fName.append(data.fName); |
| |
| if (force4f) { |
| fName.append("_4f"); |
| } |
| |
| this->setupPaint(&fPaint); |
| fPaint.setShader(MakeShader(gradType, data, tm, scale, force4f)); |
| } |
| |
| GradientBench(GradType gradType, GradData data, bool dither, bool force4f = false) |
| : fGeomType(kRect_GeomType) { |
| |
| const char *tmname = tilemodename(SkShader::kClamp_TileMode); |
| fName.printf("gradient_%s_%s", gGrads[gradType].fName, tmname); |
| fName.append(data.fName); |
| |
| if (dither) { |
| fName.appendf("_dither"); |
| } |
| |
| this->setupPaint(&fPaint); |
| fPaint.setShader(MakeShader(gradType, data, SkShader::kClamp_TileMode, 1.0f, force4f)); |
| fPaint.setDither(dither); |
| } |
| |
| protected: |
| const char* onGetName() override { |
| return fName.c_str(); |
| } |
| |
| SkIPoint onGetSize() override { |
| return SkIPoint::Make(kSize, kSize); |
| } |
| |
| void onDraw(int loops, SkCanvas* canvas) override { |
| const SkRect r = SkRect::MakeIWH(kSize, kSize); |
| |
| for (int i = 0; i < loops; i++) { |
| switch (fGeomType) { |
| case kRect_GeomType: |
| canvas->drawRect(r, fPaint); |
| break; |
| case kOval_GeomType: |
| canvas->drawOval(r, fPaint); |
| break; |
| } |
| } |
| } |
| |
| private: |
| typedef Benchmark INHERITED; |
| |
| sk_sp<SkShader> MakeShader(GradType gradType, GradData data, |
| SkShader::TileMode tm, float scale, bool force4f) { |
| const SkPoint pts[2] = { |
| { 0, 0 }, |
| { SkIntToScalar(kSize), SkIntToScalar(kSize) } |
| }; |
| |
| return gGrads[gradType].fMaker(pts, data, tm, scale, force4f); |
| } |
| |
| static const int kSize = 400; |
| |
| SkString fName; |
| SkPaint fPaint; |
| const GeomType fGeomType; |
| }; |
| |
| // 4f |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kClamp_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1], SkShader::kClamp_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2], SkShader::kClamp_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kRepeat_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1], SkShader::kRepeat_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2], SkShader::kRepeat_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kMirror_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1], SkShader::kMirror_TileMode, |
| kRect_GeomType, 1, true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2], SkShader::kMirror_TileMode, |
| kRect_GeomType, 1, true); ) |
| |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0]); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2]); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kRepeat_TileMode); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1], SkShader::kRepeat_TileMode); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2], SkShader::kRepeat_TileMode); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kMirror_TileMode); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1], SkShader::kMirror_TileMode); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2], SkShader::kMirror_TileMode); ) |
| |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0]); ) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[2]); ) |
| // Draw a radial gradient of radius 1/2 on a rectangle; half the lines should |
| // be completely pinned, the other half should pe partially pinned |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kClamp_TileMode, kRect_GeomType, 0.5f); ) |
| |
| // Draw a radial gradient on a circle of equal size; all the lines should |
| // hit the unpinned fast path (so long as GradientBench.W == H) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kClamp_TileMode, kOval_GeomType); ) |
| |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kMirror_TileMode); ) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kRepeat_TileMode); ) |
| DEF_BENCH( return new GradientBench(kSweep_GradType); ) |
| DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[2]); ) |
| DEF_BENCH( return new GradientBench(kConical_GradType); ) |
| DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[2]); ) |
| DEF_BENCH( return new GradientBench(kConicalZero_GradType); ) |
| DEF_BENCH( return new GradientBench(kConicalZero_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kConicalZero_GradType, gGradData[2]); ) |
| DEF_BENCH( return new GradientBench(kConicalOut_GradType); ) |
| DEF_BENCH( return new GradientBench(kConicalOut_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kConicalOut_GradType, gGradData[2]); ) |
| DEF_BENCH( return new GradientBench(kConicalOutZero_GradType); ) |
| DEF_BENCH( return new GradientBench(kConicalOutZero_GradType, gGradData[1]); ) |
| DEF_BENCH( return new GradientBench(kConicalOutZero_GradType, gGradData[2]); ) |
| |
| // Dithering |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[3], true); ) |
| DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[3], false); ) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[3], true); ) |
| DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[3], false); ) |
| DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[3], true); ) |
| DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[3], false); ) |
| DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[3], true); ) |
| DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[3], false); ) |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| class Gradient2Bench : public Benchmark { |
| SkString fName; |
| bool fHasAlpha; |
| |
| public: |
| Gradient2Bench(bool hasAlpha) { |
| fName.printf("gradient_create_%s", hasAlpha ? "alpha" : "opaque"); |
| fHasAlpha = hasAlpha; |
| } |
| |
| protected: |
| virtual const char* onGetName() { |
| return fName.c_str(); |
| } |
| |
| virtual void onDraw(int loops, SkCanvas* canvas) { |
| SkPaint paint; |
| this->setupPaint(&paint); |
| |
| const SkRect r = { 0, 0, SkIntToScalar(4), SkIntToScalar(4) }; |
| const SkPoint pts[] = { |
| { 0, 0 }, |
| { SkIntToScalar(100), SkIntToScalar(100) }, |
| }; |
| |
| for (int i = 0; i < loops; i++) { |
| const int gray = i % 256; |
| const int alpha = fHasAlpha ? gray : 0xFF; |
| SkColor colors[] = { |
| SK_ColorBLACK, |
| SkColorSetARGB(alpha, gray, gray, gray), |
| SK_ColorWHITE }; |
| paint.setShader(SkGradientShader::MakeLinear(pts, colors, nullptr, |
| SK_ARRAY_COUNT(colors), |
| SkShader::kClamp_TileMode)); |
| canvas->drawRect(r, paint); |
| } |
| } |
| |
| private: |
| typedef Benchmark INHERITED; |
| }; |
| |
| DEF_BENCH( return new Gradient2Bench(false); ) |
| DEF_BENCH( return new Gradient2Bench(true); ) |