blob: 52f177db8aa4e8adafd5385763416da389825fdc [file] [log] [blame]
reed@google.comdb87c962012-11-02 21:11:12 +00001/*
humper@google.coma99a92c2013-02-20 16:42:06 +00002* Copyright 2012 Google Inc.
3*
4* Use of this source code is governed by a BSD-style license that can be
5* found in the LICENSE file.
6*/
reed@google.comdb87c962012-11-02 21:11:12 +00007
8#include "gm.h"
9#include "SkBlurMaskFilter.h"
humper@google.com7c7292c2013-01-04 20:29:03 +000010#include "SkBlurMask.h"
reed@google.comdb87c962012-11-02 21:11:12 +000011#include "SkCanvas.h"
12#include "SkPath.h"
13
14#define STROKE_WIDTH SkIntToScalar(10)
15
16typedef void (*Proc)(SkCanvas*, const SkRect&, const SkPaint&);
17
18static void fill_rect(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
19 canvas->drawRect(r, p);
20}
21
reed@google.comdb87c962012-11-02 21:11:12 +000022static void draw_donut(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
23 SkRect rect;
24 SkPath path;
skia.committer@gmail.com34587162012-11-20 02:01:23 +000025
reed@google.comdb87c962012-11-02 21:11:12 +000026 rect = r;
27 rect.outset(STROKE_WIDTH/2, STROKE_WIDTH/2);
28 path.addRect(rect);
29 rect = r;
30 rect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000031
reed@google.comdb87c962012-11-02 21:11:12 +000032 path.addRect(rect);
33 path.setFillType(SkPath::kEvenOdd_FillType);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000034
reed@google.com808b70f2012-11-19 16:14:02 +000035 canvas->drawPath(path, p);
36}
reed@google.comdb87c962012-11-02 21:11:12 +000037
reed@google.com808b70f2012-11-19 16:14:02 +000038static void draw_donut_skewed(SkCanvas* canvas, const SkRect& r, const SkPaint& p) {
39 SkRect rect;
40 SkPath path;
skia.committer@gmail.com34587162012-11-20 02:01:23 +000041
reed@google.com808b70f2012-11-19 16:14:02 +000042 rect = r;
43 rect.outset(STROKE_WIDTH/2, STROKE_WIDTH/2);
44 path.addRect(rect);
45 rect = r;
46 rect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000047
reed@google.com808b70f2012-11-19 16:14:02 +000048 rect.offset(7, -7);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000049
reed@google.com808b70f2012-11-19 16:14:02 +000050 path.addRect(rect);
51 path.setFillType(SkPath::kEvenOdd_FillType);
skia.committer@gmail.com34587162012-11-20 02:01:23 +000052
reed@google.comdb87c962012-11-02 21:11:12 +000053 canvas->drawPath(path, p);
54}
55
reed@google.com53007a22012-11-26 14:39:50 +000056#include "SkGradientShader.h"
57
58typedef void (*PaintProc)(SkPaint*, SkScalar width);
59
reed@google.com57850b92012-12-17 21:20:53 +000060static const char* gBlurStyle2Name[] = {
61 "normal",
62 "solid",
63 "outer",
64 "inner"
65};
66
reed@google.comdb87c962012-11-02 21:11:12 +000067class BlurRectGM : public skiagm::GM {
humper@google.coma99a92c2013-02-20 16:42:06 +000068 SkAutoTUnref<SkMaskFilter> fMaskFilter;
69 SkString fName;
70 PaintProc fPProc;
71 SkAlpha fAlpha;
reed@google.comdb87c962012-11-02 21:11:12 +000072public:
reed@google.com57850b92012-12-17 21:20:53 +000073 BlurRectGM(const char name[], PaintProc pproc, U8CPU alpha,
74 SkBlurMaskFilter::BlurStyle bs) :
75 fMaskFilter(SkBlurMaskFilter::Create(STROKE_WIDTH/2, bs,
humper@google.coma99a92c2013-02-20 16:42:06 +000076 SkBlurMaskFilter::kHighQuality_BlurFlag))
77 , fName(name)
78 , fPProc(pproc)
79 , fAlpha(SkToU8(alpha)) {
reed@google.com57850b92012-12-17 21:20:53 +000080 fName.appendf("_%s", gBlurStyle2Name[bs]);
81 }
reed@google.comdb87c962012-11-02 21:11:12 +000082
83protected:
84 virtual SkString onShortName() {
reed@google.com53007a22012-11-26 14:39:50 +000085 return fName;
reed@google.comdb87c962012-11-02 21:11:12 +000086 }
87
88 virtual SkISize onISize() {
89 return SkISize::Make(640, 480);
90 }
91
92 virtual void onDraw(SkCanvas* canvas) {
93 canvas->translate(STROKE_WIDTH*3/2, STROKE_WIDTH*3/2);
94
reed@google.com53007a22012-11-26 14:39:50 +000095 SkRect r = { 0, 0, 250, 120 };
skia.committer@gmail.com8ccf5902012-11-27 02:01:19 +000096
reed@google.comdb87c962012-11-02 21:11:12 +000097 SkPaint paint;
98 paint.setMaskFilter(fMaskFilter);
reed@google.com53007a22012-11-26 14:39:50 +000099 if (fPProc) {
100 fPProc(&paint, r.width());
101 }
102 paint.setAlpha(fAlpha);
reed@google.comdb87c962012-11-02 21:11:12 +0000103
104 static const Proc procs[] = {
reed@google.com808b70f2012-11-19 16:14:02 +0000105 fill_rect, draw_donut, draw_donut_skewed
reed@google.comdb87c962012-11-02 21:11:12 +0000106 };
107
reed@google.comdb87c962012-11-02 21:11:12 +0000108 this->drawProcs(canvas, r, paint, false, procs, SK_ARRAY_COUNT(procs));
109 canvas->translate(r.width() * 4/3, 0);
110 this->drawProcs(canvas, r, paint, true, procs, SK_ARRAY_COUNT(procs));
111 }
112
113 virtual uint32_t onGetFlags() const { return kSkipPipe_Flag; }
114
115private:
116 void drawProcs(SkCanvas* canvas, const SkRect& r, const SkPaint& paint,
117 bool doClip, const Proc procs[], size_t procsCount) {
118 SkAutoCanvasRestore acr(canvas, true);
119 for (size_t i = 0; i < procsCount; ++i) {
120 if (doClip) {
121 SkRect clipRect(r);
122 clipRect.inset(STROKE_WIDTH/2, STROKE_WIDTH/2);
123 canvas->save();
124 canvas->clipRect(r);
125 }
126 procs[i](canvas, r, paint);
127 if (doClip) {
128 canvas->restore();
129 }
130 canvas->translate(0, r.height() * 4/3);
131 }
132 }
humper@google.coma99a92c2013-02-20 16:42:06 +0000133private:
reed@google.comdb87c962012-11-02 21:11:12 +0000134 typedef GM INHERITED;
135};
136
humper@google.com7c7292c2013-01-04 20:29:03 +0000137class BlurRectCompareGM : public skiagm::GM {
138 SkString fName;
139 unsigned int fRectWidth, fRectHeight;
reed@google.com140d7282013-01-07 20:25:04 +0000140 SkScalar fRadius;
humper@google.coma99a92c2013-02-20 16:42:06 +0000141 SkBlurMask::Style fStyle;
humper@google.com7c7292c2013-01-04 20:29:03 +0000142public:
humper@google.coma99a92c2013-02-20 16:42:06 +0000143 BlurRectCompareGM(const char name[], unsigned int rectWidth, unsigned int rectHeight, float radius, SkBlurMask::Style style)
bsalomon@google.com33cdbde2013-01-11 20:54:44 +0000144 : fName(name)
145 , fRectWidth(rectWidth)
146 , fRectHeight(rectHeight)
147 , fRadius(radius)
humper@google.coma99a92c2013-02-20 16:42:06 +0000148 , fStyle(style)
149 {}
skia.committer@gmail.com8ae714b2013-01-05 02:02:05 +0000150
humper@google.coma99a92c2013-02-20 16:42:06 +0000151 int width() const {
152 return fRectWidth;
153 }
154 int height() const {
155 return fRectHeight;
156 }
157 SkScalar radius() const {
158 return fRadius;
159 }
160 SkBlurMask::Style style() const {
161 return fStyle;
162 }
humper@google.com7c7292c2013-01-04 20:29:03 +0000163
164protected:
165 virtual SkString onShortName() {
166 return fName;
167 }
168
169 virtual SkISize onISize() {
170 return SkISize::Make(640, 480);
171 }
skia.committer@gmail.com8ae714b2013-01-05 02:02:05 +0000172
humper@google.coma99a92c2013-02-20 16:42:06 +0000173 virtual bool makeMask(SkMask *m, const SkRect&) = 0;
humper@google.com7c7292c2013-01-04 20:29:03 +0000174
175 virtual void onDraw(SkCanvas* canvas) {
humper@google.coma99a92c2013-02-20 16:42:06 +0000176 SkRect r;
177 r.setWH(SkIntToScalar(fRectWidth), SkIntToScalar(fRectHeight));
humper@google.com7c7292c2013-01-04 20:29:03 +0000178
humper@google.coma99a92c2013-02-20 16:42:06 +0000179 SkISize canvas_size = canvas->getDeviceSize();
180 int center_x = (canvas_size.fWidth - r.width())/2;
181 int center_y = (canvas_size.fHeight - r.height())/2;
humper@google.com7c7292c2013-01-04 20:29:03 +0000182
humper@google.coma99a92c2013-02-20 16:42:06 +0000183 SkMask mask;
humper@google.com7c7292c2013-01-04 20:29:03 +0000184
humper@google.coma99a92c2013-02-20 16:42:06 +0000185 if (!this->makeMask(&mask, r)) {
186 SkPaint paint;
187 r.offset( center_x, center_y );
188 canvas->drawRect(r,paint);
189 return;
190 }
191 SkAutoMaskFreeImage amfi(mask.fImage);
192
193 SkBitmap bm;
194 bm.setConfig(SkBitmap::kA8_Config, mask.fBounds.width(), mask.fBounds.height());
195 bm.setPixels(mask.fImage);
196
197 center_x = (canvas_size.fWidth - mask.fBounds.width())/2;
198 center_y = (canvas_size.fHeight - mask.fBounds.height())/2;
199
200 canvas->drawBitmap(bm, center_x, center_y, NULL);
humper@google.com7c7292c2013-01-04 20:29:03 +0000201 }
202
203 virtual uint32_t onGetFlags() const { return kSkipPipe_Flag; }
204
205private:
206 typedef GM INHERITED;
207};
208
209class BlurRectFastGM: public BlurRectCompareGM {
210public:
reed@google.com140d7282013-01-07 20:25:04 +0000211 BlurRectFastGM(const char name[], unsigned int rect_width,
humper@google.coma99a92c2013-02-20 16:42:06 +0000212 unsigned int rect_height, float blur_radius,
213 SkBlurMask::Style style) :
214 INHERITED(name, rect_width, rect_height, blur_radius, style)
215 {
216
217 }
humper@google.com7c7292c2013-01-04 20:29:03 +0000218protected:
humper@google.coma99a92c2013-02-20 16:42:06 +0000219 virtual bool makeMask(SkMask *m, const SkRect& r) SK_OVERRIDE {
220 return SkBlurMask::BlurRect(m, r, this->radius(), this->style());
reed@google.com140d7282013-01-07 20:25:04 +0000221 }
humper@google.coma99a92c2013-02-20 16:42:06 +0000222private:
223 typedef BlurRectCompareGM INHERITED;
humper@google.com7c7292c2013-01-04 20:29:03 +0000224};
225
226class BlurRectSlowGM: public BlurRectCompareGM {
227public:
humper@google.coma99a92c2013-02-20 16:42:06 +0000228 BlurRectSlowGM(const char name[], unsigned int rect_width, unsigned int rect_height,
229 float blur_radius, SkBlurMask::Style style) :
230 INHERITED(name, rect_width, rect_height, blur_radius, style)
231 {
232
233 }
humper@google.com7c7292c2013-01-04 20:29:03 +0000234protected:
humper@google.coma99a92c2013-02-20 16:42:06 +0000235 virtual bool makeMask(SkMask *m, const SkRect& r) SK_OVERRIDE {
reed@google.com140d7282013-01-07 20:25:04 +0000236 SkMask src;
237 r.roundOut(&src.fBounds);
238 src.fBounds.offset(-src.fBounds.fLeft, -src.fBounds.fTop); // move to origin
239 src.fFormat = SkMask::kA8_Format;
240 src.fRowBytes = src.fBounds.width();
humper@google.coma99a92c2013-02-20 16:42:06 +0000241 src.fImage = SkMask::AllocImage(src.computeTotalImageSize());
bsalomon@google.com33cdbde2013-01-11 20:54:44 +0000242 SkAutoMaskFreeImage amfi(src.fImage);
humper@google.com7c7292c2013-01-04 20:29:03 +0000243
bsalomon@google.com33cdbde2013-01-11 20:54:44 +0000244 memset(src.fImage, 0xff, src.computeTotalImageSize());
humper@google.com7c7292c2013-01-04 20:29:03 +0000245
humper@google.coma99a92c2013-02-20 16:42:06 +0000246 return SkBlurMask::BlurSeparable(m, src, this->radius(), this->style(), this->getQuality());
reed@google.com140d7282013-01-07 20:25:04 +0000247 }
humper@google.coma99a92c2013-02-20 16:42:06 +0000248
249 virtual SkBlurMask::Quality getQuality() {
250 return SkBlurMask::kHigh_Quality;
251 }
252private:
253 typedef BlurRectCompareGM INHERITED;
254};
255
256class BlurRectSlowLowGM: public BlurRectSlowGM {
257public:
258 BlurRectSlowLowGM(const char name[], unsigned int rectWidth, unsigned int rectHeight,
259 float blurRadius, SkBlurMask::Style style) :
260 INHERITED(name, rectWidth, rectHeight, blurRadius, style)
261 {
262
263 }
264protected:
265 virtual SkBlurMask::Quality getQuality() SK_OVERRIDE {
266 return SkBlurMask::kLow_Quality;
267 }
268private:
269 typedef BlurRectSlowGM INHERITED;
270};
271
272class BlurRectGroundTruthGM: public BlurRectCompareGM {
273public:
274 BlurRectGroundTruthGM(const char name[], unsigned int rectWidth, unsigned int rectHeight,
275 float blurRadius, SkBlurMask::Style style) :
276 INHERITED(name, rectWidth, rectHeight, blurRadius, style)
277 {
278
279 }
280protected:
281 virtual bool makeMask(SkMask *m, const SkRect& r) SK_OVERRIDE {
282 SkMask src;
283 r.roundOut(&src.fBounds);
284 src.fBounds.offset(-src.fBounds.fLeft, -src.fBounds.fTop); // move to origin
285 src.fFormat = SkMask::kA8_Format;
286 src.fRowBytes = src.fBounds.width();
287 src.fImage = SkMask::AllocImage(src.computeTotalImageSize());
288 SkAutoMaskFreeImage amfi(src.fImage);
289
290 memset(src.fImage, 0xff, src.computeTotalImageSize());
291
292 return SkBlurMask::BlurGroundTruth(m, src, this->radius(), this->style());
293 }
294
295 virtual SkBlurMask::Quality getQuality() {
296 return SkBlurMask::kHigh_Quality;
297 }
298private:
299 typedef BlurRectCompareGM INHERITED;
humper@google.com7c7292c2013-01-04 20:29:03 +0000300};
301
302
reed@google.comdb87c962012-11-02 21:11:12 +0000303//////////////////////////////////////////////////////////////////////////////
304
reed@google.com57850b92012-12-17 21:20:53 +0000305DEF_GM(return new BlurRectGM("blurrect", NULL, 0xFF, SkBlurMaskFilter::kNormal_BlurStyle);)
306DEF_GM(return new BlurRectGM("blurrect", NULL, 0xFF, SkBlurMaskFilter::kSolid_BlurStyle);)
307DEF_GM(return new BlurRectGM("blurrect", NULL, 0xFF, SkBlurMaskFilter::kOuter_BlurStyle);)
308DEF_GM(return new BlurRectGM("blurrect", NULL, 0xFF, SkBlurMaskFilter::kInner_BlurStyle);)
309
humper@google.coma99a92c2013-02-20 16:42:06 +0000310// regular size rects, blurs should be small enough not to completely overlap.
reed@google.comdb87c962012-11-02 21:11:12 +0000311
humper@google.coma99a92c2013-02-20 16:42:06 +0000312DEF_GM(return new BlurRectFastGM( "blurrect_25_100_2_normal_fast", 25, 100, 2, SkBlurMask::kNormal_Style);)
313DEF_GM(return new BlurRectFastGM("blurrect_25_100_20_normal_fast", 25, 100, 20, SkBlurMask::kNormal_Style);)
314DEF_GM(return new BlurRectSlowGM( "blurrect_25_100_2_normal_slow", 25, 100, 2, SkBlurMask::kNormal_Style);)
315DEF_GM(return new BlurRectSlowGM("blurrect_25_100_20_normal_slow", 25, 100, 20, SkBlurMask::kNormal_Style);)
316DEF_GM(return new BlurRectFastGM( "blurrect_25_100_2_inner_fast", 25, 100, 2, SkBlurMask::kInner_Style);)
317DEF_GM(return new BlurRectFastGM("blurrect_25_100_20_inner_fast", 25, 100, 20, SkBlurMask::kInner_Style);)
318DEF_GM(return new BlurRectSlowGM( "blurrect_25_100_2_inner_slow", 25, 100, 2, SkBlurMask::kInner_Style);)
319DEF_GM(return new BlurRectSlowGM("blurrect_25_100_20_inner_slow", 25, 100, 20, SkBlurMask::kInner_Style);)
320DEF_GM(return new BlurRectFastGM( "blurrect_25_100_2_outer_fast", 25, 100, 2, SkBlurMask::kOuter_Style);)
321DEF_GM(return new BlurRectFastGM("blurrect_25_100_20_outer_fast", 25, 100, 20, SkBlurMask::kOuter_Style);)
322DEF_GM(return new BlurRectSlowGM( "blurrect_25_100_2_outer_slow", 25, 100, 2, SkBlurMask::kOuter_Style);)
323DEF_GM(return new BlurRectSlowGM("blurrect_25_100_20_outer_slow", 25, 100, 20, SkBlurMask::kOuter_Style);)
324
325// skinny tall rects, blurs overlap in X but not y
326
327DEF_GM(return new BlurRectFastGM( "blurrect_5_100_2_normal_fast", 5, 100, 2 , SkBlurMask::kNormal_Style);)
328DEF_GM(return new BlurRectFastGM("blurrect_5_100_20_normal_fast", 5, 100, 20, SkBlurMask::kNormal_Style);)
329DEF_GM(return new BlurRectSlowGM( "blurrect_5_100_2_normal_slow", 5, 100, 2 , SkBlurMask::kNormal_Style);)
330DEF_GM(return new BlurRectSlowGM("blurrect_5_100_20_normal_slow", 5, 100, 20, SkBlurMask::kNormal_Style);)
331DEF_GM(return new BlurRectFastGM( "blurrect_5_100_2_inner_fast", 5, 100, 2 , SkBlurMask::kInner_Style);)
332DEF_GM(return new BlurRectFastGM("blurrect_5_100_20_inner_fast", 5, 100, 20, SkBlurMask::kInner_Style);)
333DEF_GM(return new BlurRectSlowGM( "blurrect_5_100_2_inner_slow", 5, 100, 2 , SkBlurMask::kInner_Style);)
334DEF_GM(return new BlurRectSlowGM("blurrect_5_100_20_inner_slow", 5, 100, 20, SkBlurMask::kInner_Style);)
335DEF_GM(return new BlurRectFastGM( "blurrect_5_100_2_outer_fast", 5, 100, 2 , SkBlurMask::kOuter_Style);)
336DEF_GM(return new BlurRectFastGM("blurrect_5_100_20_outer_fast", 5, 100, 20, SkBlurMask::kOuter_Style);)
337DEF_GM(return new BlurRectSlowGM( "blurrect_5_100_2_outer_slow", 5, 100, 2 , SkBlurMask::kOuter_Style);)
338DEF_GM(return new BlurRectSlowGM("blurrect_5_100_20_outer_slow", 5, 100, 20, SkBlurMask::kOuter_Style);)
339
340// tiny rects, blurs overlap in X and Y
341
342DEF_GM(return new BlurRectFastGM( "blurrect_5_5_2_normal_fast", 5, 5, 2 , SkBlurMask::kNormal_Style);)
343DEF_GM(return new BlurRectFastGM("blurrect_5_5_20_normal_fast", 5, 5, 20, SkBlurMask::kNormal_Style);)
344DEF_GM(return new BlurRectSlowGM( "blurrect_5_5_2_normal_slow", 5, 5, 2 , SkBlurMask::kNormal_Style);)
345DEF_GM(return new BlurRectSlowGM("blurrect_5_5_20_normal_slow", 5, 5, 20, SkBlurMask::kNormal_Style);)
346DEF_GM(return new BlurRectFastGM( "blurrect_5_5_2_inner_fast", 5, 5, 2 , SkBlurMask::kInner_Style);)
347DEF_GM(return new BlurRectFastGM("blurrect_5_5_20_inner_fast", 5, 5, 20, SkBlurMask::kInner_Style);)
348DEF_GM(return new BlurRectSlowGM( "blurrect_5_5_2_inner_slow", 5, 5, 2 , SkBlurMask::kInner_Style);)
349DEF_GM(return new BlurRectSlowGM("blurrect_5_5_20_inner_slow", 5, 5, 20, SkBlurMask::kInner_Style);)
350DEF_GM(return new BlurRectFastGM( "blurrect_5_5_2_outer_fast", 5, 5, 2 , SkBlurMask::kOuter_Style);)
351DEF_GM(return new BlurRectFastGM("blurrect_5_5_20_outer_fast", 5, 5, 20, SkBlurMask::kOuter_Style);)
352DEF_GM(return new BlurRectSlowGM( "blurrect_5_5_2_outer_slow", 5, 5, 2 , SkBlurMask::kOuter_Style);)
353DEF_GM(return new BlurRectSlowGM("blurrect_5_5_20_outer_slow", 5, 5, 20, SkBlurMask::kOuter_Style);)
354
355
356#if 0
357// dont' need to GM the gaussian convolution; it's slow and intended
358// as a ground truth comparison only. Leaving these here in case we
359// ever want to turn these back on for debugging reasons.
360DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_1_simple", 25, 100, 1);)
361DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_2_simple", 25, 100, 2);)
362DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_3_simple", 25, 100, 3);)
363DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_4_simple", 25, 100, 4);)
364DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_5_simple", 25, 100, 5);)
365DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_6_simple", 25, 100, 6);)
366DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_7_simple", 25, 100, 7);)
367DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_8_simple", 25, 100, 8);)
368DEF_GM(return new BlurRectGroundTruthGM( "blurrect_25_100_9_simple", 25, 100, 9);)
369DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_10_simple", 25, 100, 10);)
370DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_11_simple", 25, 100, 11);)
371DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_12_simple", 25, 100, 12);)
372DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_13_simple", 25, 100, 13);)
373DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_14_simple", 25, 100, 14);)
374DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_15_simple", 25, 100, 15);)
375DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_16_simple", 25, 100, 16);)
376DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_17_simple", 25, 100, 17);)
377DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_18_simple", 25, 100, 18);)
378DEF_GM(return new BlurRectGroundTruthGM("blurrect_25_100_19_simple", 25, 100, 19);)
379#endif