blob: cad32466fc0a45dc38de93b9f788d641bb0b616b [file] [log] [blame]
kjlubick43195932016-04-05 12:48:47 -07001/*
2 * Copyright 2013 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 */
7#include "Fuzz.h"
8#include "Sk1DPathEffect.h"
9#include "Sk2DPathEffect.h"
10#include "SkAlphaThresholdFilter.h"
11#include "SkArcToPathEffect.h"
12#include "SkBlurImageFilter.h"
13#include "SkBlurMaskFilter.h"
14#include "SkCanvas.h"
15#include "SkColorCubeFilter.h"
16#include "SkColorFilter.h"
17#include "SkColorFilterImageFilter.h"
18#include "SkColorMatrixFilter.h"
19#include "SkComposeImageFilter.h"
20#include "SkCornerPathEffect.h"
21#include "SkDashPathEffect.h"
22#include "SkData.h"
23#include "SkDiscretePathEffect.h"
24#include "SkDisplacementMapEffect.h"
25#include "SkDropShadowImageFilter.h"
26#include "SkEmbossMaskFilter.h"
27#include "SkFlattenableSerialization.h"
28#include "SkImageSource.h"
29#include "SkLayerRasterizer.h"
30#include "SkLightingImageFilter.h"
31#include "SkLumaColorFilter.h"
32#include "SkMagnifierImageFilter.h"
33#include "SkMatrixConvolutionImageFilter.h"
34#include "SkMergeImageFilter.h"
35#include "SkMorphologyImageFilter.h"
36#include "SkOffsetImageFilter.h"
37#include "SkPaintImageFilter.h"
38#include "SkPerlinNoiseShader.h"
39#include "SkPictureImageFilter.h"
40#include "SkPictureRecorder.h"
41#include "SkPoint3.h"
42#include "SkRandom.h"
43#include "SkTableColorFilter.h"
44#include "SkTestImageFilters.h"
45#include "SkTileImageFilter.h"
46#include "SkTypeface.h"
47#include "SkXfermodeImageFilter.h"
48#include <stdio.h>
49#include <time.h>
50
51#define SK_ADD_RANDOM_BIT_FLIPS
52
53static Fuzz* fuzz;
54static const int kBitmapSize = 24;
55
56static bool return_large = false;
57static bool return_undef = false;
58
59static int R(float x) {
60 return (int)floor(SkScalarToFloat(fuzz->nextF1()) * x);
61}
62
63#if defined _WIN32
64#pragma warning ( push )
65// we are intentionally causing an overflow here
66// (warning C4756: overflow in constant arithmetic)
67#pragma warning ( disable : 4756 )
68#endif
69
70static float huge() {
71 double d = 1e100;
72 float f = (float)d;
73 return f;
74}
75
76#if defined _WIN32
77#pragma warning ( pop )
78#endif
79
80static float make_number(bool positiveOnly) {
81 float f = positiveOnly ? 1.0f : 0.0f;
82 float v = f;
83 int sel;
84
85 if (return_large) sel = R(6); else sel = R(4);
86 if (!return_undef && sel == 0) sel = 1;
87
88 if (R(2) == 1) v = (float)(R(100)+f); else
89
90 switch (sel) {
91 case 0: break;
92 case 1: v = f; break;
93 case 2: v = 0.000001f; break;
94 case 3: v = 10000.0f; break;
95 case 4: v = 2000000000.0f; break;
96 case 5: v = huge(); break;
97 }
98
99 if (!positiveOnly && (R(4) == 1)) v = -v;
100 return v;
101}
102
103static SkScalar make_scalar(bool positiveOnly = false) {
104 return make_number(positiveOnly);
105}
106
107static SkString make_string() {
108 int length = R(1000);
109 SkString str(length);
110 for (int i = 0; i < length; ++i) {
111 str[i] = static_cast<char>(R(256));
112 }
113 return str;
114}
115
116static SkString make_font_name() {
117 int sel = R(8);
118
119 switch(sel) {
120 case 0: return SkString("Courier New");
121 case 1: return SkString("Helvetica");
122 case 2: return SkString("monospace");
123 case 3: return SkString("sans-serif");
124 case 4: return SkString("serif");
125 case 5: return SkString("Times");
126 case 6: return SkString("Times New Roman");
127 case 7:
128 default:
129 return make_string();
130 }
131}
132
133static bool make_bool() {
134 return R(2) == 1;
135}
136
137static SkRect make_rect() {
138 return SkRect::MakeWH(SkIntToScalar(R(static_cast<float>(kBitmapSize))),
139 SkIntToScalar(R(static_cast<float>(kBitmapSize))));
140}
141
142static SkRegion make_region() {
143 SkIRect iRegion = SkIRect::MakeXYWH(R(static_cast<float>(kBitmapSize)),
144 R(static_cast<float>(kBitmapSize)),
145 R(static_cast<float>(kBitmapSize)),
146 R(static_cast<float>(kBitmapSize)));
147 return SkRegion(iRegion);
148}
149
150static SkMatrix make_matrix() {
151 SkMatrix m;
152 for (int i = 0; i < 9; ++i) {
153 m[i] = make_scalar();
154 }
155 return m;
156}
157
158static SkXfermode::Mode make_xfermode() {
159 return static_cast<SkXfermode::Mode>(R(SkXfermode::kLastMode+1));
160}
161
162static SkPaint::Align make_paint_align() {
163 return static_cast<SkPaint::Align>(R(SkPaint::kRight_Align+1));
164}
165
166static SkPaint::Hinting make_paint_hinting() {
167 return static_cast<SkPaint::Hinting>(R(SkPaint::kFull_Hinting+1));
168}
169
170static SkPaint::Style make_paint_style() {
171 return static_cast<SkPaint::Style>(R(SkPaint::kStrokeAndFill_Style+1));
172}
173
174static SkPaint::Cap make_paint_cap() {
175 return static_cast<SkPaint::Cap>(R(SkPaint::kDefault_Cap+1));
176}
177
178static SkPaint::Join make_paint_join() {
179 return static_cast<SkPaint::Join>(R(SkPaint::kDefault_Join+1));
180}
181
182static SkPaint::TextEncoding make_paint_text_encoding() {
183 return static_cast<SkPaint::TextEncoding>(R(SkPaint::kGlyphID_TextEncoding+1));
184}
185
186static SkBlurStyle make_blur_style() {
187 return static_cast<SkBlurStyle>(R(kLastEnum_SkBlurStyle+1));
188}
189
190static SkBlurMaskFilter::BlurFlags make_blur_mask_filter_flag() {
191 return static_cast<SkBlurMaskFilter::BlurFlags>(R(SkBlurMaskFilter::kAll_BlurFlag+1));
192}
193
194static SkFilterQuality make_filter_quality() {
195 return static_cast<SkFilterQuality>(R(kHigh_SkFilterQuality+1));
196}
197
198static SkTypeface::Style make_typeface_style() {
199 return static_cast<SkTypeface::Style>(R(SkTypeface::kBoldItalic+1));
200}
201
202static SkPath1DPathEffect::Style make_path_1d_path_effect_style() {
203 return static_cast<SkPath1DPathEffect::Style>(R((int)SkPath1DPathEffect::kLastEnum_Style + 1));
204}
205
206static SkColor make_color() {
207 return (R(2) == 1) ? 0xFFC0F0A0 : 0xFF000090;
208}
209
210static SkDropShadowImageFilter::ShadowMode make_shadow_mode() {
211 return (R(2) == 1) ? SkDropShadowImageFilter::kDrawShadowAndForeground_ShadowMode :
212 SkDropShadowImageFilter::kDrawShadowOnly_ShadowMode;
213}
214
215static SkPoint3 make_point() {
216 return SkPoint3::Make(make_scalar(), make_scalar(), make_scalar(true));
217}
218
219static SkDisplacementMapEffect::ChannelSelectorType make_channel_selector_type() {
220 return static_cast<SkDisplacementMapEffect::ChannelSelectorType>(R(4)+1);
221}
222
223static bool valid_for_raster_canvas(const SkImageInfo& info) {
224 switch (info.colorType()) {
225 case kAlpha_8_SkColorType:
226 case kRGB_565_SkColorType:
227 return true;
228 case kN32_SkColorType:
229 return kPremul_SkAlphaType == info.alphaType() ||
230 kOpaque_SkAlphaType == info.alphaType();
231 default:
232 break;
233 }
234 return false;
235}
236
237static SkColorType rand_colortype() {
238 return (SkColorType)R(kLastEnum_SkColorType + 1);
239}
240
241static void rand_bitmap_for_canvas(SkBitmap* bitmap) {
242 SkImageInfo info;
243 do {
244 info = SkImageInfo::Make(kBitmapSize, kBitmapSize, rand_colortype(),
245 kPremul_SkAlphaType);
246 } while (!valid_for_raster_canvas(info) || !bitmap->tryAllocPixels(info));
247}
248
249static void make_g_bitmap(SkBitmap& bitmap) {
250 rand_bitmap_for_canvas(&bitmap);
251
252 SkCanvas canvas(bitmap);
253 canvas.clear(0x00000000);
254 SkPaint paint;
255 paint.setAntiAlias(true);
256 paint.setColor(0xFF884422);
257 paint.setTextSize(SkIntToScalar(kBitmapSize/2));
258 const char* str = "g";
259 canvas.drawText(str, strlen(str), SkIntToScalar(kBitmapSize/8),
260 SkIntToScalar(kBitmapSize/4), paint);
261}
262
263static void make_checkerboard_bitmap(SkBitmap& bitmap) {
264 rand_bitmap_for_canvas(&bitmap);
265
266 SkCanvas canvas(bitmap);
267 canvas.clear(0x00000000);
268 SkPaint darkPaint;
269 darkPaint.setColor(0xFF804020);
270 SkPaint lightPaint;
271 lightPaint.setColor(0xFF244484);
272 const int i = kBitmapSize / 8;
273 const SkScalar f = SkIntToScalar(i);
274 for (int y = 0; y < kBitmapSize; y += i) {
275 for (int x = 0; x < kBitmapSize; x += i) {
276 canvas.save();
277 canvas.translate(SkIntToScalar(x), SkIntToScalar(y));
278 canvas.drawRect(SkRect::MakeXYWH(0, 0, f, f), darkPaint);
279 canvas.drawRect(SkRect::MakeXYWH(f, 0, f, f), lightPaint);
280 canvas.drawRect(SkRect::MakeXYWH(0, f, f, f), lightPaint);
281 canvas.drawRect(SkRect::MakeXYWH(f, f, f, f), darkPaint);
282 canvas.restore();
283 }
284 }
285}
286
287static const SkBitmap& make_bitmap() {
288 static SkBitmap bitmap[2];
289 static bool initialized = false;
290 if (!initialized) {
291 make_g_bitmap(bitmap[0]);
292 make_checkerboard_bitmap(bitmap[1]);
293 initialized = true;
294 }
295 return bitmap[R(2)];
296}
297
298static sk_sp<SkData> make_3Dlut(int* cubeDimension, bool invR, bool invG, bool invB) {
299 int size = 4 << R(5);
300 auto data = SkData::MakeUninitialized(sizeof(SkColor) * size * size * size);
301 SkColor* pixels = (SkColor*)(data->writable_data());
302 SkAutoTMalloc<uint8_t> lutMemory(size);
303 SkAutoTMalloc<uint8_t> invLutMemory(size);
304 uint8_t* lut = lutMemory.get();
305 uint8_t* invLut = invLutMemory.get();
306 const int maxIndex = size - 1;
307 for (int i = 0; i < size; i++) {
308 lut[i] = (i * 255) / maxIndex;
309 invLut[i] = ((maxIndex - i) * 255) / maxIndex;
310 }
311 for (int r = 0; r < size; ++r) {
312 for (int g = 0; g < size; ++g) {
313 for (int b = 0; b < size; ++b) {
314 pixels[(size * ((size * b) + g)) + r] = SkColorSetARGB(0xFF,
315 invR ? invLut[r] : lut[r],
316 invG ? invLut[g] : lut[g],
317 invB ? invLut[b] : lut[b]);
318 }
319 }
320 }
321 if (cubeDimension) {
322 *cubeDimension = size;
323 }
324 return data;
325}
326
327static void drawSomething(SkCanvas* canvas) {
328 SkPaint paint;
329
330 canvas->save();
331 canvas->scale(0.5f, 0.5f);
332 canvas->drawBitmap(make_bitmap(), 0, 0, nullptr);
333 canvas->restore();
334
335 paint.setAntiAlias(true);
336
337 paint.setColor(SK_ColorRED);
338 canvas->drawCircle(SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/3), paint);
339 paint.setColor(SK_ColorBLACK);
340 paint.setTextSize(SkIntToScalar(kBitmapSize/3));
341 canvas->drawText("Picture", 7, SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/4), paint);
342}
343
344static void rand_color_table(uint8_t* table) {
345 for (int i = 0; i < 256; ++i) {
346 table[i] = R(256);
347 }
348}
349
350static sk_sp<SkColorFilter> make_color_filter() {
351 switch (R(6)) {
352 case 0: {
353 SkScalar array[20];
354 for (int i = 0; i < 20; ++i) {
355 array[i] = make_scalar();
356 }
357 return SkColorFilter::MakeMatrixFilterRowMajor255(array);
358 }
359 case 1:
360 return SkLumaColorFilter::Make();
361 case 2: {
362 uint8_t tableA[256];
363 uint8_t tableR[256];
364 uint8_t tableG[256];
365 uint8_t tableB[256];
366 rand_color_table(tableA);
367 rand_color_table(tableR);
368 rand_color_table(tableG);
369 rand_color_table(tableB);
370 return SkTableColorFilter::MakeARGB(tableA, tableR, tableG, tableB);
371 }
372 case 3:
373 return SkColorFilter::MakeModeFilter(make_color(), make_xfermode());
374 case 4:
375 return SkColorMatrixFilter::MakeLightingFilter(make_color(), make_color());
376 case 5:
377 default:
378 break;
379 }
380 return nullptr;
381}
382
383static SkPath make_path() {
384 SkPath path;
385 int numOps = R(30);
386 for (int i = 0; i < numOps; ++i) {
387 switch (R(6)) {
388 case 0:
389 path.moveTo(make_scalar(), make_scalar());
390 break;
391 case 1:
392 path.lineTo(make_scalar(), make_scalar());
393 break;
394 case 2:
395 path.quadTo(make_scalar(), make_scalar(), make_scalar(), make_scalar());
396 break;
397 case 3:
398 path.conicTo(make_scalar(), make_scalar(), make_scalar(), make_scalar(), make_scalar());
399 break;
400 case 4:
401 path.cubicTo(make_scalar(), make_scalar(), make_scalar(),
402 make_scalar(), make_scalar(), make_scalar());
403 break;
404 case 5:
405 default:
406 path.arcTo(make_scalar(), make_scalar(), make_scalar(), make_scalar(), make_scalar());
407 break;
408
409 }
410 }
411 path.close();
412 return path;
413}
414
415static sk_sp<SkPathEffect> make_path_effect(bool canBeNull = true) {
416 sk_sp<SkPathEffect> pathEffect;
417 if (canBeNull && (R(3) == 1)) { return pathEffect; }
418
419 switch (R(9)) {
420 case 0:
421 pathEffect = SkArcToPathEffect::Make(make_scalar(true));
422 break;
423 case 1:
424 pathEffect = SkComposePathEffect::Make(make_path_effect(false),
425 make_path_effect(false));
426 break;
427 case 2:
428 pathEffect = SkCornerPathEffect::Make(make_scalar());
429 break;
430 case 3: {
431 int count = R(10);
432 SkScalar intervals[10];
433 for (int i = 0; i < count; ++i) {
434 intervals[i] = make_scalar();
435 }
436 pathEffect = SkDashPathEffect::Make(intervals, count, make_scalar());
437 break;
438 }
439 case 4:
440 pathEffect = SkDiscretePathEffect::Make(make_scalar(), make_scalar());
441 break;
442 case 5:
443 pathEffect = SkPath1DPathEffect::Make(make_path(), make_scalar(), make_scalar(),
444 make_path_1d_path_effect_style());
445 break;
446 case 6:
447 pathEffect = SkLine2DPathEffect::Make(make_scalar(), make_matrix());
448 break;
449 case 7:
450 pathEffect = SkPath2DPathEffect::Make(make_matrix(), make_path());
451 break;
452 case 8:
453 default:
454 pathEffect = SkSumPathEffect::Make(make_path_effect(false),
455 make_path_effect(false));
456 break;
457 }
458 return pathEffect;
459}
460
461static sk_sp<SkMaskFilter> make_mask_filter() {
462 sk_sp<SkMaskFilter> maskFilter;
463 switch (R(3)) {
464 case 0:
465 maskFilter = SkBlurMaskFilter::Make(make_blur_style(), make_scalar(),
466 make_blur_mask_filter_flag());
467 case 1: {
468 SkEmbossMaskFilter::Light light;
469 for (int i = 0; i < 3; ++i) {
470 light.fDirection[i] = make_scalar();
471 }
472 light.fPad = R(65536);
473 light.fAmbient = R(256);
474 light.fSpecular = R(256);
475 maskFilter = SkEmbossMaskFilter::Make(make_scalar(), light);
476 }
477 case 2:
478 default:
479 break;
480 }
481 return maskFilter;
482}
483
484static sk_sp<SkImageFilter> make_image_filter(bool canBeNull = true);
485
486static SkPaint make_paint() {
487 SkPaint paint;
488 paint.setHinting(make_paint_hinting());
489 paint.setAntiAlias(make_bool());
490 paint.setDither(make_bool());
491 paint.setLinearText(make_bool());
492 paint.setSubpixelText(make_bool());
493 paint.setLCDRenderText(make_bool());
494 paint.setEmbeddedBitmapText(make_bool());
495 paint.setAutohinted(make_bool());
496 paint.setVerticalText(make_bool());
497 paint.setUnderlineText(make_bool());
498 paint.setStrikeThruText(make_bool());
499 paint.setFakeBoldText(make_bool());
500 paint.setDevKernText(make_bool());
501 paint.setFilterQuality(make_filter_quality());
502 paint.setStyle(make_paint_style());
503 paint.setColor(make_color());
504 paint.setStrokeWidth(make_scalar());
505 paint.setStrokeMiter(make_scalar());
506 paint.setStrokeCap(make_paint_cap());
507 paint.setStrokeJoin(make_paint_join());
508 paint.setColorFilter(make_color_filter());
509 paint.setXfermodeMode(make_xfermode());
510 paint.setPathEffect(make_path_effect());
511 paint.setMaskFilter(make_mask_filter());
512
513 if (false) {
514 // our validating buffer does not support typefaces yet, so skip this for now
515 SkAutoTUnref<SkTypeface> typeface(
516 SkTypeface::CreateFromName(make_font_name().c_str(), make_typeface_style()));
517 paint.setTypeface(typeface);
518 }
519
520 SkLayerRasterizer::Builder rasterizerBuilder;
521 SkPaint paintForRasterizer;
522 if (R(2) == 1) {
523 paintForRasterizer = make_paint();
524 }
525 rasterizerBuilder.addLayer(paintForRasterizer);
526 paint.setRasterizer(rasterizerBuilder.detach());
527 paint.setImageFilter(make_image_filter());
528 sk_sp<SkData> data(make_3Dlut(nullptr, make_bool(), make_bool(), make_bool()));
529 paint.setTextAlign(make_paint_align());
530 paint.setTextSize(make_scalar());
531 paint.setTextScaleX(make_scalar());
532 paint.setTextSkewX(make_scalar());
533 paint.setTextEncoding(make_paint_text_encoding());
534 return paint;
535}
536
537static sk_sp<SkImageFilter> make_image_filter(bool canBeNull) {
538 sk_sp<SkImageFilter> filter;
539
540 // Add a 1 in 3 chance to get a nullptr input
541 if (canBeNull && (R(3) == 1)) {
542 return filter;
543 }
544
545 enum { ALPHA_THRESHOLD, MERGE, COLOR, LUT3D, BLUR, MAGNIFIER,
546 DOWN_SAMPLE, XFERMODE, OFFSET, MATRIX, MATRIX_CONVOLUTION, COMPOSE,
547 DISTANT_LIGHT, POINT_LIGHT, SPOT_LIGHT, NOISE, DROP_SHADOW,
548 MORPHOLOGY, BITMAP, DISPLACE, TILE, PICTURE, PAINT, NUM_FILTERS };
549
550 switch (R(NUM_FILTERS)) {
551 case ALPHA_THRESHOLD:
552 filter = SkAlphaThresholdFilter::Make(make_region(),
553 make_scalar(),
554 make_scalar(),
robertphillipsef6a47b2016-04-08 08:01:20 -0700555 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700556 break;
557 case MERGE:
558 filter = SkMergeImageFilter::Make(make_image_filter(),
559 make_image_filter(),
560 make_xfermode());
561 break;
562 case COLOR: {
563 sk_sp<SkColorFilter> cf(make_color_filter());
564 filter = cf ? SkColorFilterImageFilter::Make(std::move(cf), make_image_filter())
565 : nullptr;
566 break;
567 }
568 case LUT3D: {
569 int cubeDimension;
570 sk_sp<SkData> lut3D(make_3Dlut(&cubeDimension, (R(2) == 1), (R(2) == 1), (R(2) == 1)));
571 sk_sp<SkColorFilter> cf(SkColorCubeFilter::Make(std::move(lut3D), cubeDimension));
572 filter = cf ? SkColorFilterImageFilter::Make(std::move(cf), make_image_filter())
573 : nullptr;
574 break;
575 }
576 case BLUR:
577 filter = SkBlurImageFilter::Make(make_scalar(true),
578 make_scalar(true),
579 make_image_filter());
580 break;
581 case MAGNIFIER:
robertphillips11171f32016-04-07 07:34:15 -0700582 filter = SkMagnifierImageFilter::Make(make_rect(),
kjlubick9fd07e52016-04-12 12:02:59 -0700583 make_scalar(true),
robertphillips11171f32016-04-07 07:34:15 -0700584 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700585 break;
586 case DOWN_SAMPLE:
robertphillipsfc0f44a2016-04-06 07:44:01 -0700587 filter = SkDownSampleImageFilter::Make(make_scalar(), make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700588 break;
589 case XFERMODE:
590 filter = SkXfermodeImageFilter::Make(SkXfermode::Make(make_xfermode()),
591 make_image_filter(),
592 make_image_filter(),
593 nullptr);
594 break;
595 case OFFSET:
596 filter = SkOffsetImageFilter::Make(make_scalar(), make_scalar(), make_image_filter());
597 break;
robertphillipsae8c9332016-04-05 15:09:00 -0700598 case MATRIX:
599 filter = SkImageFilter::MakeMatrixFilter(make_matrix(),
600 (SkFilterQuality)R(4),
601 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700602 break;
kjlubick43195932016-04-05 12:48:47 -0700603 case MATRIX_CONVOLUTION: {
604 SkImageFilter::CropRect cropR(SkRect::MakeWH(SkIntToScalar(kBitmapSize),
605 SkIntToScalar(kBitmapSize)));
606 SkISize size = SkISize::Make(R(10)+1, R(10)+1);
607 int arraySize = size.width() * size.height();
608 SkTArray<SkScalar> kernel(arraySize);
609 for (int i = 0; i < arraySize; ++i) {
610 kernel.push_back() = make_scalar();
611 }
612 SkIPoint kernelOffset = SkIPoint::Make(R(SkIntToScalar(size.width())),
613 R(SkIntToScalar(size.height())));
kjlubick43195932016-04-05 12:48:47 -0700614
robertphillipsef6a47b2016-04-08 08:01:20 -0700615 filter = SkMatrixConvolutionImageFilter::Make(size,
616 kernel.begin(),
617 make_scalar(),
618 make_scalar(),
619 kernelOffset,
620 (SkMatrixConvolutionImageFilter::TileMode)R(3),
621 R(2) == 1,
622 make_image_filter(),
623 &cropR);
kjlubick43195932016-04-05 12:48:47 -0700624 break;
625 }
626 case COMPOSE:
627 filter = SkComposeImageFilter::Make(make_image_filter(), make_image_filter());
628 break;
robertphillips12fa47d2016-04-08 16:28:09 -0700629 case DISTANT_LIGHT:
kjlubick43195932016-04-05 12:48:47 -0700630 filter = (R(2) == 1)
robertphillips12fa47d2016-04-08 16:28:09 -0700631 ? SkLightingImageFilter::MakeDistantLitDiffuse(make_point(), make_color(),
632 make_scalar(), make_scalar(),
633 make_image_filter())
634 : SkLightingImageFilter::MakeDistantLitSpecular(make_point(), make_color(),
635 make_scalar(), make_scalar(),
636 SkIntToScalar(R(10)),
637 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700638 break;
robertphillips12fa47d2016-04-08 16:28:09 -0700639 case POINT_LIGHT:
kjlubick43195932016-04-05 12:48:47 -0700640 filter = (R(2) == 1)
robertphillips12fa47d2016-04-08 16:28:09 -0700641 ? SkLightingImageFilter::MakePointLitDiffuse(make_point(), make_color(),
642 make_scalar(), make_scalar(),
643 make_image_filter())
644 : SkLightingImageFilter::MakePointLitSpecular(make_point(), make_color(),
645 make_scalar(), make_scalar(),
646 SkIntToScalar(R(10)),
647 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700648 break;
robertphillips12fa47d2016-04-08 16:28:09 -0700649 case SPOT_LIGHT:
kjlubick43195932016-04-05 12:48:47 -0700650 filter = (R(2) == 1)
robertphillips12fa47d2016-04-08 16:28:09 -0700651 ? SkLightingImageFilter::MakeSpotLitDiffuse(SkPoint3::Make(0, 0, 0),
652 make_point(), make_scalar(),
653 make_scalar(), make_color(),
654 make_scalar(), make_scalar(),
655 make_image_filter())
656 : SkLightingImageFilter::MakeSpotLitSpecular(SkPoint3::Make(0, 0, 0),
657 make_point(), make_scalar(),
658 make_scalar(), make_color(),
659 make_scalar(), make_scalar(),
660 SkIntToScalar(R(10)),
661 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700662 break;
kjlubick43195932016-04-05 12:48:47 -0700663 case NOISE: {
robertphillips12fa47d2016-04-08 16:28:09 -0700664 sk_sp<SkShader> shader((R(2) == 1)
665 ? SkPerlinNoiseShader::MakeFractalNoise(make_scalar(true), make_scalar(true),
666 R(10.0f), make_scalar())
667 : SkPerlinNoiseShader::MakeTurbulence(make_scalar(true), make_scalar(true),
668 R(10.0f), make_scalar()));
kjlubick43195932016-04-05 12:48:47 -0700669 SkPaint paint;
670 paint.setShader(shader);
671 SkImageFilter::CropRect cropR(SkRect::MakeWH(SkIntToScalar(kBitmapSize),
672 SkIntToScalar(kBitmapSize)));
673 filter = SkPaintImageFilter::Make(paint, &cropR);
674 break;
675 }
robertphillipsc4169122016-04-06 08:40:59 -0700676 case DROP_SHADOW:
677 filter = SkDropShadowImageFilter::Make(make_scalar(),
678 make_scalar(),
679 make_scalar(true),
680 make_scalar(true),
681 make_color(),
682 make_shadow_mode(),
683 make_image_filter(),
684 nullptr);
kjlubick43195932016-04-05 12:48:47 -0700685 break;
kjlubick43195932016-04-05 12:48:47 -0700686 case MORPHOLOGY:
687 if (R(2) == 1) {
688 filter = SkDilateImageFilter::Make(R(static_cast<float>(kBitmapSize)),
689 R(static_cast<float>(kBitmapSize)),
690 make_image_filter());
691 } else {
692 filter = SkErodeImageFilter::Make(R(static_cast<float>(kBitmapSize)),
693 R(static_cast<float>(kBitmapSize)),
694 make_image_filter());
695 }
696 break;
697 case BITMAP: {
698 sk_sp<SkImage> image(SkImage::MakeFromBitmap(make_bitmap()));
699 if (R(2) == 1) {
700 filter = SkImageSource::Make(std::move(image),
701 make_rect(),
702 make_rect(),
703 kHigh_SkFilterQuality);
704 } else {
705 filter = SkImageSource::Make(std::move(image));
706 }
707 break;
708 }
robertphillipsbfe11fc2016-04-15 07:17:36 -0700709 case DISPLACE:
710 filter = SkDisplacementMapEffect::Make(make_channel_selector_type(),
711 make_channel_selector_type(),
712 make_scalar(),
713 make_image_filter(false),
714 make_image_filter());
kjlubick43195932016-04-05 12:48:47 -0700715 break;
robertphillips534c2702016-04-15 07:57:40 -0700716 case TILE:
717 filter = SkTileImageFilter::Make(make_rect(), make_rect(), make_image_filter(false));
kjlubick43195932016-04-05 12:48:47 -0700718 break;
kjlubick43195932016-04-05 12:48:47 -0700719 case PICTURE: {
720 SkRTreeFactory factory;
721 SkPictureRecorder recorder;
722 SkCanvas* recordingCanvas = recorder.beginRecording(SkIntToScalar(kBitmapSize),
723 SkIntToScalar(kBitmapSize),
724 &factory, 0);
725 drawSomething(recordingCanvas);
726 sk_sp<SkPicture> pict(recorder.finishRecordingAsPicture());
727 filter = SkPictureImageFilter::Make(pict, make_rect());
728 break;
729 }
730 case PAINT: {
731 SkImageFilter::CropRect cropR(make_rect());
732 filter = SkPaintImageFilter::Make(make_paint(), &cropR);
733 break;
734 }
735 default:
736 break;
737 }
738 return (filter || canBeNull) ? filter : make_image_filter(canBeNull);
739}
740
741static SkImageFilter* make_serialized_image_filter() {
742 sk_sp<SkImageFilter> filter(make_image_filter(false));
743 SkAutoTUnref<SkData> data(SkValidatingSerializeFlattenable(filter.get()));
744 const unsigned char* ptr = static_cast<const unsigned char*>(data->data());
745 size_t len = data->size();
746#ifdef SK_ADD_RANDOM_BIT_FLIPS
747 unsigned char* p = const_cast<unsigned char*>(ptr);
748 for (size_t i = 0; i < len; ++i, ++p) {
749 if (R(250) == 1) { // 0.4% of the time, flip a bit or byte
750 if (R(10) == 1) { // Then 10% of the time, change a whole byte
751 switch(R(3)) {
752 case 0:
753 *p ^= 0xFF; // Flip entire byte
754 break;
755 case 1:
756 *p = 0xFF; // Set all bits to 1
757 break;
758 case 2:
759 *p = 0x00; // Set all bits to 0
760 break;
761 }
762 } else {
763 *p ^= (1 << R(8));
764 }
765 }
766 }
767#endif // SK_ADD_RANDOM_BIT_FLIPS
768 SkFlattenable* flattenable = SkValidatingDeserializeFlattenable(ptr, len,
769 SkImageFilter::GetFlattenableType());
770 return static_cast<SkImageFilter*>(flattenable);
771}
772
773static void drawClippedBitmap(SkCanvas* canvas, int x, int y, const SkPaint& paint) {
774 canvas->save();
775 canvas->clipRect(SkRect::MakeXYWH(SkIntToScalar(x), SkIntToScalar(y),
776 SkIntToScalar(kBitmapSize), SkIntToScalar(kBitmapSize)));
777 canvas->drawBitmap(make_bitmap(), SkIntToScalar(x), SkIntToScalar(y), &paint);
778 canvas->restore();
779}
780
781DEF_FUZZ(SerializedImageFilter, f) {
782 fuzz = f;
783 SkImageFilter* filter = make_serialized_image_filter();
784
785 SkPaint paint;
786 SkSafeUnref(paint.setImageFilter(filter));
787 SkBitmap bitmap;
788 SkCanvas canvas(bitmap);
789 drawClippedBitmap(&canvas, 0, 0, paint);
790}