BlurRectBench.cpp revision e396455d2d60ddf8e625b5037254f3c09fbcdcf5
15c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)/* 25c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) * Copyright 2013 Google Inc. 35c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) * 45c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) * Use of this source code is governed by a BSD-style license that can be 55c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) * found in the LICENSE file. 65c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) */ 75c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 85c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkBenchmark.h" 95c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkCanvas.h" 105c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkPaint.h" 115c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkRandom.h" 125c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkShader.h" 135c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkString.h" 145c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#include "SkBlurMask.h" 155c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 165c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#define SMALL SkIntToScalar(2) 175c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#define REAL 1.5f 185c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)static const SkScalar kMedium = SkIntToScalar(5); 195c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#define BIG SkIntToScalar(10) 205c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)static const SkScalar kMedBig = SkIntToScalar(20); 215c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)#define REALBIG 30.5f 225c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 2302772c6a72f1ee0b226341a4f4439970c29fc861Ben Murdochclass BlurRectBench: public SkBenchmark { 245c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) int fLoopCount; 255c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) SkScalar fRadius; 265c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) SkString fName; 27a854de003a23bf3c7f95ec0f8154ada64092ff5cTorne (Richard Coles) 285c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)public: 295c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) BlurRectBench(SkScalar rad) { 305c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) fRadius = rad; 31a854de003a23bf3c7f95ec0f8154ada64092ff5cTorne (Richard Coles) 325c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) if (fRadius > SkIntToScalar(25)) { 335c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) fLoopCount = 100; 345c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } else if (fRadius > SkIntToScalar(5)) { 355c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) fLoopCount = 1000; 365c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } else { 375c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) fLoopCount = 10000; 385c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } 395c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } 405c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 415c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles)protected: 425c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) virtual const char* onGetName() { 435c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) return fName.c_str(); 445c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } 45a854de003a23bf3c7f95ec0f8154ada64092ff5cTorne (Richard Coles) 465c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) SkScalar radius() const { 475c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) return fRadius; 485c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } 495c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 505c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) void setName(const SkString& name) { 515c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) fName = name; 525c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) } 535c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 545c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) virtual void onDraw(const int loops, SkCanvas*) { 555c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) SkPaint paint; 565c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) this->setupPaint(&paint); 575c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) 585c87bf8b86a7c82ef50fb7a89697d8e02e2553beTorne (Richard Coles) paint.setAntiAlias(true); 59 60 SkScalar pad = fRadius*3/2 + SK_Scalar1; 61 SkRect r = SkRect::MakeWH(2 * pad + SK_Scalar1, 2 * pad + SK_Scalar1); 62 63 preBenchSetup(r); 64 65 for (int i = 0; i < loops; i++) { 66 makeBlurryRect(r); 67 } 68 } 69 70 virtual void makeBlurryRect(const SkRect&) = 0; 71 virtual void preBenchSetup(const SkRect&) {} 72private: 73 typedef SkBenchmark INHERITED; 74}; 75 76 77class BlurRectDirectBench: public BlurRectBench { 78 public: 79 BlurRectDirectBench(SkScalar rad) : INHERITED(rad) { 80 SkString name; 81 82 if (SkScalarFraction(rad) != 0) { 83 name.printf("blurrect_direct_%.2f", SkScalarToFloat(rad)); 84 } else { 85 name.printf("blurrect_direct_%d", SkScalarRoundToInt(rad)); 86 } 87 88 this->setName(name); 89 } 90protected: 91 virtual void makeBlurryRect(const SkRect& r) SK_OVERRIDE { 92 SkMask mask; 93 SkBlurMask::BlurRect(SkBlurMask::ConvertRadiusToSigma(this->radius()), 94 &mask, r, kNormal_SkBlurStyle); 95 SkMask::FreeImage(mask.fImage); 96 } 97private: 98 typedef BlurRectBench INHERITED; 99}; 100 101class BlurRectSeparableBench: public BlurRectBench { 102 103public: 104 BlurRectSeparableBench(SkScalar rad) : INHERITED(rad) { 105 fSrcMask.fImage = NULL; 106 } 107 108 ~BlurRectSeparableBench() { 109 SkMask::FreeImage(fSrcMask.fImage); 110 } 111 112protected: 113 virtual void preBenchSetup(const SkRect& r) SK_OVERRIDE { 114 SkMask::FreeImage(fSrcMask.fImage); 115 116 r.roundOut(&fSrcMask.fBounds); 117 fSrcMask.fFormat = SkMask::kA8_Format; 118 fSrcMask.fRowBytes = fSrcMask.fBounds.width(); 119 fSrcMask.fImage = SkMask::AllocImage(fSrcMask.computeTotalImageSize()); 120 121 memset(fSrcMask.fImage, 0xff, fSrcMask.computeTotalImageSize()); 122 } 123 124 SkMask fSrcMask; 125private: 126 typedef BlurRectBench INHERITED; 127}; 128 129class BlurRectBoxFilterBench: public BlurRectSeparableBench { 130public: 131 BlurRectBoxFilterBench(SkScalar rad) : INHERITED(rad) { 132 SkString name; 133 134 if (SkScalarFraction(rad) != 0) { 135 name.printf("blurrect_boxfilter_%.2f", SkScalarToFloat(rad)); 136 } else { 137 name.printf("blurrect_boxfilter_%d", SkScalarRoundToInt(rad)); 138 } 139 140 this->setName(name); 141 } 142 143protected: 144 145 virtual void makeBlurryRect(const SkRect&) SK_OVERRIDE { 146 SkMask mask; 147 mask.fImage = NULL; 148 SkBlurMask::BoxBlur(&mask, fSrcMask, SkBlurMask::ConvertRadiusToSigma(this->radius()), 149 kNormal_SkBlurStyle, kHigh_SkBlurQuality); 150 SkMask::FreeImage(mask.fImage); 151 } 152private: 153 typedef BlurRectSeparableBench INHERITED; 154}; 155 156class BlurRectGaussianBench: public BlurRectSeparableBench { 157public: 158 BlurRectGaussianBench(SkScalar rad) : INHERITED(rad) { 159 SkString name; 160 161 if (SkScalarFraction(rad) != 0) { 162 name.printf("blurrect_gaussian_%.2f", SkScalarToFloat(rad)); 163 } else { 164 name.printf("blurrect_gaussian_%d", SkScalarRoundToInt(rad)); 165 } 166 167 this->setName(name); 168 } 169 170protected: 171 172 virtual void makeBlurryRect(const SkRect&) SK_OVERRIDE { 173 SkMask mask; 174 mask.fImage = NULL; 175 SkBlurMask::BlurGroundTruth(SkBlurMask::ConvertRadiusToSigma(this->radius()), 176 &mask, fSrcMask, kNormal_SkBlurStyle); 177 SkMask::FreeImage(mask.fImage); 178 } 179private: 180 typedef BlurRectSeparableBench INHERITED; 181}; 182 183DEF_BENCH(return new BlurRectBoxFilterBench(SMALL);) 184DEF_BENCH(return new BlurRectBoxFilterBench(BIG);) 185DEF_BENCH(return new BlurRectBoxFilterBench(REALBIG);) 186DEF_BENCH(return new BlurRectBoxFilterBench(REAL);) 187DEF_BENCH(return new BlurRectGaussianBench(SMALL);) 188DEF_BENCH(return new BlurRectGaussianBench(BIG);) 189DEF_BENCH(return new BlurRectGaussianBench(REALBIG);) 190DEF_BENCH(return new BlurRectGaussianBench(REAL);) 191DEF_BENCH(return new BlurRectDirectBench(SMALL);) 192DEF_BENCH(return new BlurRectDirectBench(BIG);) 193DEF_BENCH(return new BlurRectDirectBench(REALBIG);) 194DEF_BENCH(return new BlurRectDirectBench(REAL);) 195 196DEF_BENCH(return new BlurRectDirectBench(kMedium);) 197DEF_BENCH(return new BlurRectDirectBench(kMedBig);) 198 199DEF_BENCH(return new BlurRectBoxFilterBench(kMedium);) 200DEF_BENCH(return new BlurRectBoxFilterBench(kMedBig);) 201 202#if 0 203// disable Gaussian benchmarks; the algorithm works well enough 204// and serves as a baseline for ground truth, but it's too slow 205// to use in production for non-trivial radii, so no real point 206// in having the bots benchmark it all the time. 207 208DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(1));) 209DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(2));) 210DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(3));) 211DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(4));) 212DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(5));) 213DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(6));) 214DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(7));) 215DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(8));) 216DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(9));) 217DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(10));) 218DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(11));) 219DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(12));) 220DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(13));) 221DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(14));) 222DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(15));) 223DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(16));) 224DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(17));) 225DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(18));) 226DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(19));) 227DEF_BENCH(return new BlurRectGaussianBench(SkIntToScalar(20));) 228#endif 229