/* * Copyright 2013 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "Fuzz.h" #include "Sk1DPathEffect.h" #include "Sk2DPathEffect.h" #include "SkAlphaThresholdFilter.h" #include "SkArcToPathEffect.h" #include "SkBlurImageFilter.h" #include "SkBlurMaskFilter.h" #include "SkCanvas.h" #include "SkColorFilter.h" #include "SkColorFilterImageFilter.h" #include "SkColorMatrixFilter.h" #include "SkComposeImageFilter.h" #include "SkCornerPathEffect.h" #include "SkDashPathEffect.h" #include "SkData.h" #include "SkDiscretePathEffect.h" #include "SkDisplacementMapEffect.h" #include "SkDropShadowImageFilter.h" #include "SkEmbossMaskFilter.h" #include "SkFlattenableSerialization.h" #include "SkImageSource.h" #include "SkLayerRasterizer.h" #include "SkLightingImageFilter.h" #include "SkLumaColorFilter.h" #include "SkMagnifierImageFilter.h" #include "SkMatrixConvolutionImageFilter.h" #include "SkMergeImageFilter.h" #include "SkMorphologyImageFilter.h" #include "SkOffsetImageFilter.h" #include "SkPaintImageFilter.h" #include "SkPerlinNoiseShader.h" #include "SkPictureImageFilter.h" #include "SkPictureRecorder.h" #include "SkPoint3.h" #include "SkRandom.h" #include "SkTableColorFilter.h" #include "SkTileImageFilter.h" #include "SkTypeface.h" #include "SkXfermodeImageFilter.h" #include <cmath> #include <stdio.h> #include <time.h> #define SK_ADD_RANDOM_BIT_FLIPS static Fuzz* fuzz; static const int kBitmapSize = 24; // There should be no more than one make_* used as a function argument. static bool make_bool() { bool b; fuzz->next(&b); return b; } static float make_number(bool positiveOnly) { float f; fuzz->next(&f); if (positiveOnly) { return std::abs(f); } return f; } static SkString make_string() { int length; fuzz->nextRange(&length, 0, 1000); SkString str(length); for (int i = 0; i < length; ++i) { char c; fuzz->nextRange(&c, 0, 255); str[i] = c; } return str; } static SkString make_font_name() { int sel; fuzz->nextRange(&sel, 0, 7); switch(sel) { case 0: return SkString("Courier New"); case 1: return SkString("Helvetica"); case 2: return SkString("monospace"); case 3: return SkString("sans-serif"); case 4: return SkString("serif"); case 5: return SkString("Times"); case 6: return SkString("Times New Roman"); case 7: default: return make_string(); } } static SkRect make_rect() { SkScalar w, h; fuzz->nextRange(&w, 0.0f, (float) kBitmapSize-1); fuzz->nextRange(&h, 0.0f, (float) kBitmapSize-1); return SkRect::MakeWH(w, h); } static SkRegion make_region() { int32_t x, y, w, h; fuzz->nextRange(&x, 0, kBitmapSize-1); fuzz->nextRange(&y, 0, kBitmapSize-1); fuzz->nextRange(&w, 0, kBitmapSize-1); fuzz->nextRange(&h, 0, kBitmapSize-1); SkIRect iRegion = SkIRect::MakeXYWH(x,y,w,h); return SkRegion(iRegion); } static void init_matrix(SkMatrix* m) { SkScalar mat[9]; fuzz->nextN(mat, 9); m->set9(mat); } static SkBlendMode make_blendmode() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkBlendMode::kLastMode); return static_cast<SkBlendMode>(i); } static SkPaint::Align make_paint_align() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kRight_Align); return static_cast<SkPaint::Align>(i); } static SkPaint::Hinting make_paint_hinting() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kFull_Hinting); return static_cast<SkPaint::Hinting>(i); } static SkPaint::Style make_paint_style() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kStrokeAndFill_Style); return static_cast<SkPaint::Style>(i); } static SkPaint::Cap make_paint_cap() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kDefault_Cap); return static_cast<SkPaint::Cap>(i); } static SkPaint::Join make_paint_join() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kDefault_Join); return static_cast<SkPaint::Join>(i); } static SkPaint::TextEncoding make_paint_text_encoding() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPaint::kGlyphID_TextEncoding); return static_cast<SkPaint::TextEncoding>(i); } static SkBlurStyle make_blur_style() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)kLastEnum_SkBlurStyle); return static_cast<SkBlurStyle>(i); } static SkBlurMaskFilter::BlurFlags make_blur_mask_filter_flag() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkBlurMaskFilter::kAll_BlurFlag); return static_cast<SkBlurMaskFilter::BlurFlags>(i); } static SkFilterQuality make_filter_quality() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)kHigh_SkFilterQuality); return static_cast<SkFilterQuality>(i); } static SkFontStyle make_typeface_style() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkTypeface::kBoldItalic); return SkFontStyle::FromOldStyle(i); } static SkPath1DPathEffect::Style make_path_1d_path_effect_style() { uint8_t i; fuzz->nextRange(&i, 0, (uint8_t)SkPath1DPathEffect::kLastEnum_Style); return static_cast<SkPath1DPathEffect::Style>(i); } static SkColor make_color() { return make_bool() ? 0xFFC0F0A0 : 0xFF000090; } static SkDropShadowImageFilter::ShadowMode make_shadow_mode() { return make_bool() ? SkDropShadowImageFilter::kDrawShadowAndForeground_ShadowMode : SkDropShadowImageFilter::kDrawShadowOnly_ShadowMode; } static SkPoint3 make_point() { SkScalar a, b, c; fuzz->next(&a, &b, &c); c = std::abs(c); return SkPoint3::Make(a, b, c); } static SkDisplacementMapEffect::ChannelSelectorType make_channel_selector_type() { uint8_t i; fuzz->nextRange(&i, 1, (uint8_t)SkDisplacementMapEffect::kA_ChannelSelectorType); return static_cast<SkDisplacementMapEffect::ChannelSelectorType>(i); } static SkColorType rand_colortype() { uint8_t i; fuzz->nextRange(&i, 0, kLastEnum_SkColorType); return (SkColorType) i; } static void rand_bitmap_for_canvas(SkBitmap* bitmap) { SkImageInfo info = SkImageInfo::Make(kBitmapSize, kBitmapSize, rand_colortype(), kPremul_SkAlphaType); if (!bitmap->tryAllocPixels(info)){ SkDebugf("Bitmap not allocated\n"); } } static void make_g_bitmap(SkBitmap& bitmap) { rand_bitmap_for_canvas(&bitmap); SkCanvas canvas(bitmap); canvas.clear(0x00000000); SkPaint paint; paint.setAntiAlias(true); paint.setColor(0xFF884422); paint.setTextSize(SkIntToScalar(kBitmapSize/2)); const char* str = "g"; canvas.drawText(str, strlen(str), SkIntToScalar(kBitmapSize/8), SkIntToScalar(kBitmapSize/4), paint); } static void make_checkerboard_bitmap(SkBitmap& bitmap) { rand_bitmap_for_canvas(&bitmap); SkCanvas canvas(bitmap); canvas.clear(0x00000000); SkPaint darkPaint; darkPaint.setColor(0xFF804020); SkPaint lightPaint; lightPaint.setColor(0xFF244484); const int i = kBitmapSize / 8; const SkScalar f = SkIntToScalar(i); for (int y = 0; y < kBitmapSize; y += i) { for (int x = 0; x < kBitmapSize; x += i) { canvas.save(); canvas.translate(SkIntToScalar(x), SkIntToScalar(y)); canvas.drawRect(SkRect::MakeXYWH(0, 0, f, f), darkPaint); canvas.drawRect(SkRect::MakeXYWH(f, 0, f, f), lightPaint); canvas.drawRect(SkRect::MakeXYWH(0, f, f, f), lightPaint); canvas.drawRect(SkRect::MakeXYWH(f, f, f, f), darkPaint); canvas.restore(); } } } static const SkBitmap& make_bitmap() { static SkBitmap bitmap[2]; static bool initialized = false; if (!initialized) { make_g_bitmap(bitmap[0]); make_checkerboard_bitmap(bitmap[1]); initialized = true; } uint8_t i; fuzz->nextRange(&i, 0, 1); return bitmap[i]; } static sk_sp<SkData> make_3Dlut(int* cubeDimension, bool invR, bool invG, bool invB) { uint8_t shift; fuzz->nextRange(&shift, 0, 4); int size = 4 << shift; auto data = SkData::MakeUninitialized(sizeof(SkColor) * size * size * size); SkColor* pixels = (SkColor*)(data->writable_data()); SkAutoTMalloc<uint8_t> lutMemory(size); SkAutoTMalloc<uint8_t> invLutMemory(size); uint8_t* lut = lutMemory.get(); uint8_t* invLut = invLutMemory.get(); const int maxIndex = size - 1; for (int i = 0; i < size; i++) { lut[i] = (i * 255) / maxIndex; invLut[i] = ((maxIndex - i) * 255) / maxIndex; } for (int r = 0; r < size; ++r) { for (int g = 0; g < size; ++g) { for (int b = 0; b < size; ++b) { pixels[(size * ((size * b) + g)) + r] = SkColorSetARGB(0xFF, invR ? invLut[r] : lut[r], invG ? invLut[g] : lut[g], invB ? invLut[b] : lut[b]); } } } if (cubeDimension) { *cubeDimension = size; } return data; } static void drawSomething(SkCanvas* canvas) { SkPaint paint; canvas->save(); canvas->scale(0.5f, 0.5f); canvas->drawBitmap(make_bitmap(), 0, 0, nullptr); canvas->restore(); paint.setAntiAlias(true); paint.setColor(SK_ColorRED); canvas->drawCircle(SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/3), paint); paint.setColor(SK_ColorBLACK); paint.setTextSize(SkIntToScalar(kBitmapSize/3)); canvas->drawText("Picture", 7, SkIntToScalar(kBitmapSize/2), SkIntToScalar(kBitmapSize/4), paint); } static sk_sp<SkColorFilter> make_color_filter() { uint8_t s; fuzz->nextRange(&s, 0, 5); switch (s) { case 0: { SkScalar array[20]; fuzz->nextN(array, 20); return SkColorFilter::MakeMatrixFilterRowMajor255(array); } case 1: return SkLumaColorFilter::Make(); case 2: { uint8_t tableA[256]; uint8_t tableR[256]; uint8_t tableG[256]; uint8_t tableB[256]; fuzz->nextN(tableA, 256); fuzz->nextN(tableR, 256); fuzz->nextN(tableG, 256); fuzz->nextN(tableB, 256); return SkTableColorFilter::MakeARGB(tableA, tableR, tableG, tableB); } case 3: { SkColor c = make_color(); SkBlendMode mode = make_blendmode(); return SkColorFilter::MakeModeFilter(c, mode); } case 4: { SkColor a = make_color(); SkColor b = make_color(); return SkColorMatrixFilter::MakeLightingFilter(a, b); } case 5: default: break; } return nullptr; } static SkPath make_path() { SkPath path; uint8_t numOps; fuzz->nextRange(&numOps, 0, 30); for (uint8_t i = 0; i < numOps; ++i) { uint8_t op; fuzz->nextRange(&op, 0, 5); SkScalar a, b, c, d, e, f; switch (op) { case 0: fuzz->next(&a, &b); path.moveTo(a, b); break; case 1: fuzz->next(&a, &b); path.lineTo(a, b); break; case 2: fuzz->next(&a, &b, &c, &d); path.quadTo(a, b, c, d); break; case 3: fuzz->next(&a, &b, &c, &d, &e); path.conicTo(a, b, c, d, e); break; case 4: fuzz->next(&a, &b, &c, &d, &e, &f); path.cubicTo(a, b, c, d, e, f); break; case 5: default: fuzz->next(&a, &b, &c, &d, &e); path.arcTo(a, b, c, d, e); break; } } path.close(); return path; } static sk_sp<SkPathEffect> make_path_effect(bool canBeNull = true) { sk_sp<SkPathEffect> pathEffect; uint8_t s; fuzz->nextRange(&s, 0, 2); if (canBeNull && s == 0) { return pathEffect; } fuzz->nextRange(&s, 0, 8); switch (s) { case 0: { SkScalar a = make_number(true); pathEffect = SkArcToPathEffect::Make(a); break; } case 1: { sk_sp<SkPathEffect> a = make_path_effect(false); sk_sp<SkPathEffect> b = make_path_effect(false); pathEffect = SkPathEffect::MakeCompose(a, b); break; } case 2: { SkScalar a = make_number(false); pathEffect = SkCornerPathEffect::Make(a); break; } case 3: { uint8_t count; fuzz->nextRange(&count, 0, 9); SkScalar intervals[10]; fuzz->nextN(intervals, 10); SkScalar a = make_number(false); pathEffect = SkDashPathEffect::Make(intervals, count, a); break; } case 4: { SkScalar a, b; fuzz->next(&a, &b); pathEffect = SkDiscretePathEffect::Make(a, b); break; } case 5: { SkPath path = make_path(); SkScalar a, b; fuzz->next(&a, &b); SkPath1DPathEffect::Style style = make_path_1d_path_effect_style(); pathEffect = SkPath1DPathEffect::Make(path, a, b, style); break; } case 6: { SkScalar a = make_number(false); SkMatrix m; init_matrix(&m); pathEffect = SkLine2DPathEffect::Make(a, m); break; } case 7: { SkPath path = make_path(); SkMatrix m; init_matrix(&m); pathEffect = SkPath2DPathEffect::Make(m, path); break; } case 8: default: { sk_sp<SkPathEffect> a = make_path_effect(false); sk_sp<SkPathEffect> b = make_path_effect(false); pathEffect = SkPathEffect::MakeCompose(a, b); break; } } return pathEffect; } static sk_sp<SkMaskFilter> make_mask_filter() { sk_sp<SkMaskFilter> maskFilter; uint8_t s; fuzz->nextRange(&s, 0, 2); switch (s) { case 0: { SkBlurStyle blur = make_blur_style(); SkScalar a = make_number(false); SkBlurMaskFilter::BlurFlags flags = make_blur_mask_filter_flag(); maskFilter = SkBlurMaskFilter::Make(blur, a, flags); break; } case 1: { SkScalar a = make_number(false); SkEmbossMaskFilter::Light light; fuzz->nextN(light.fDirection, 3); fuzz->nextRange(&light.fPad, 0, 65535); fuzz->nextRange(&light.fAmbient, 0, 255); fuzz->nextRange(&light.fSpecular, 0, 255); maskFilter = SkEmbossMaskFilter::Make(a, light); break; } case 2: default: break; } return maskFilter; } static sk_sp<SkImageFilter> make_image_filter(bool canBeNull = true); static SkPaint make_paint() { SkPaint paint; if (fuzz->exhausted()) { return paint; } paint.setHinting(make_paint_hinting()); paint.setAntiAlias(make_bool()); paint.setDither(make_bool()); paint.setLinearText(make_bool()); paint.setSubpixelText(make_bool()); paint.setLCDRenderText(make_bool()); paint.setEmbeddedBitmapText(make_bool()); paint.setAutohinted(make_bool()); paint.setVerticalText(make_bool()); paint.setFakeBoldText(make_bool()); paint.setDevKernText(make_bool()); paint.setFilterQuality(make_filter_quality()); paint.setStyle(make_paint_style()); paint.setColor(make_color()); paint.setStrokeWidth(make_number(false)); paint.setStrokeMiter(make_number(false)); paint.setStrokeCap(make_paint_cap()); paint.setStrokeJoin(make_paint_join()); paint.setColorFilter(make_color_filter()); paint.setBlendMode(make_blendmode()); paint.setPathEffect(make_path_effect()); paint.setMaskFilter(make_mask_filter()); if (false) { // our validating buffer does not support typefaces yet, so skip this for now paint.setTypeface(SkTypeface::MakeFromName(make_font_name().c_str(),make_typeface_style())); } SkLayerRasterizer::Builder rasterizerBuilder; SkPaint paintForRasterizer; if (make_bool()) { paintForRasterizer = make_paint(); } rasterizerBuilder.addLayer(paintForRasterizer); paint.setRasterizer(rasterizerBuilder.detach()); paint.setImageFilter(make_image_filter()); bool a, b, c; fuzz->next(&a, &b, &c); sk_sp<SkData> data(make_3Dlut(nullptr, a, b, c)); paint.setTextAlign(make_paint_align()); SkScalar d, e, f; fuzz->next(&d, &e, &f); paint.setTextSize(d); paint.setTextScaleX(e); paint.setTextSkewX(f); paint.setTextEncoding(make_paint_text_encoding()); return paint; } static sk_sp<SkImageFilter> make_image_filter(bool canBeNull) { sk_sp<SkImageFilter> filter; // Add a 1 in 3 chance to get a nullptr input uint8_t i; fuzz->nextRange(&i, 0, 2); if (fuzz->exhausted() || (canBeNull && i == 1)) { return filter; } enum { ALPHA_THRESHOLD, MERGE, COLOR, BLUR, MAGNIFIER, BLENDMODE, OFFSET, MATRIX, MATRIX_CONVOLUTION, COMPOSE, DISTANT_LIGHT, POINT_LIGHT, SPOT_LIGHT, NOISE, DROP_SHADOW, MORPHOLOGY, BITMAP, DISPLACE, TILE, PICTURE, PAINT, NUM_FILTERS }; uint8_t s; fuzz->nextRange(&s, 0, NUM_FILTERS - 1); switch (s) { case ALPHA_THRESHOLD: { SkRegion reg = make_region(); SkScalar innerMin, outerMax; fuzz->next(&innerMin, &outerMax); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkAlphaThresholdFilter::Make(reg, innerMin, outerMax, fil); break; } case MERGE: { sk_sp<SkImageFilter> filA = make_image_filter(); sk_sp<SkImageFilter> filB = make_image_filter(); SkBlendMode blend = make_blendmode(); filter = SkMergeImageFilter::Make(filA, filB, blend); break; } case COLOR: { sk_sp<SkColorFilter> cf(make_color_filter()); filter = cf ? SkColorFilterImageFilter::Make(std::move(cf), make_image_filter()) : nullptr; break; } case BLUR: { SkScalar sX = make_number(true); SkScalar sY = make_number(true); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkBlurImageFilter::Make(sX, sY, fil); break; } case MAGNIFIER: { SkRect rect = make_rect(); SkScalar inset = make_number(true); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkMagnifierImageFilter::Make(rect, inset, fil); break; } case BLENDMODE: { SkBlendMode mode = make_blendmode(); sk_sp<SkImageFilter> filA = make_image_filter(); sk_sp<SkImageFilter> filB = make_image_filter(); filter = SkXfermodeImageFilter::Make(mode, filA, filB, nullptr); break; } case OFFSET: { SkScalar dx, dy; fuzz->next(&dx, &dy); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkOffsetImageFilter::Make(dx, dy, fil); break; } case MATRIX: { SkMatrix m; init_matrix(&m); int qual; fuzz->nextRange(&qual, 0, SkFilterQuality::kLast_SkFilterQuality - 1); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkImageFilter::MakeMatrixFilter(m, (SkFilterQuality)qual, fil); break; } case MATRIX_CONVOLUTION: { SkImageFilter::CropRect cropR(SkRect::MakeWH(SkIntToScalar(kBitmapSize), SkIntToScalar(kBitmapSize))); int w, h; fuzz->nextRange(&w, 1, 10); fuzz->nextRange(&h, 1, 10); SkISize size = SkISize::Make(w, h); int arraySize = size.width() * size.height(); SkTArray<SkScalar> kernel(arraySize); for (int i = 0; i < arraySize; ++i) { kernel.push_back() = make_number(false); } fuzz->nextRange(&w, 0, size.width() - 1); fuzz->nextRange(&h, 0, size.height() - 1); SkIPoint kernelOffset = SkIPoint::Make(w, h); int mode; fuzz->nextRange(&mode, 0, SkMatrixConvolutionImageFilter::kMax_TileMode - 1); bool convolveAlpha = make_bool(); SkScalar gain, bias; fuzz->next(&gain, &bias); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkMatrixConvolutionImageFilter::Make(size, kernel.begin(), gain, bias, kernelOffset, (SkMatrixConvolutionImageFilter::TileMode)mode, convolveAlpha, fil, &cropR); break; } case COMPOSE: { sk_sp<SkImageFilter> filA = make_image_filter(); sk_sp<SkImageFilter> filB = make_image_filter(); filter = SkComposeImageFilter::Make(filA, filB); break; } case DISTANT_LIGHT: { SkPoint3 p = make_point(); SkColor c = make_color(); SkScalar ss, kd; fuzz->next(&ss, &kd); int shininess; fuzz->nextRange(&shininess, 0, 9); sk_sp<SkImageFilter> fil = make_image_filter(); filter = make_bool() ? SkLightingImageFilter::MakeDistantLitDiffuse(p, c, ss, kd, fil) : SkLightingImageFilter::MakeDistantLitSpecular(p, c, ss, kd, shininess, fil); break; } case POINT_LIGHT: { SkPoint3 p = make_point(); SkColor c = make_color(); SkScalar ss, kd; fuzz->next(&ss, &kd); int shininess; fuzz->nextRange(&shininess, 0, 9); sk_sp<SkImageFilter> fil = make_image_filter(); filter = make_bool() ? SkLightingImageFilter::MakePointLitDiffuse(p, c, ss, kd, fil) : SkLightingImageFilter::MakePointLitSpecular(p, c, ss, kd, shininess, fil); break; } case SPOT_LIGHT: { SkPoint3 p = make_point(); SkColor c = make_color(); SkScalar se, ca, ss, kd; fuzz->next(&se, &ca, &ss, &kd); int shininess; fuzz->nextRange(&shininess, 0, 9); sk_sp<SkImageFilter> fil = make_image_filter(); filter = make_bool() ? SkLightingImageFilter::MakeSpotLitDiffuse(SkPoint3::Make(0, 0, 0), p, se, ca, c, ss, kd, fil) : SkLightingImageFilter::MakeSpotLitSpecular(SkPoint3::Make(0, 0, 0), p, se, ca, c, ss, kd, shininess, fil); break; } case NOISE: { SkScalar bfx = make_number(true); SkScalar bfy = make_number(true); SkScalar seed = make_number(false); int octaves; fuzz->nextRange(&octaves, 0, 9); sk_sp<SkShader> shader(make_bool() ? SkPerlinNoiseShader::MakeFractalNoise(bfx, bfy, octaves, seed) : SkPerlinNoiseShader::MakeTurbulence(bfx, bfy, octaves, seed)); SkPaint paint; paint.setShader(shader); SkImageFilter::CropRect cropR(SkRect::MakeWH(SkIntToScalar(kBitmapSize), SkIntToScalar(kBitmapSize))); filter = SkPaintImageFilter::Make(paint, &cropR); break; } case DROP_SHADOW: { SkScalar dx, dy, sx, sy; fuzz->next(&dx, &dy); sx = make_number(true); sy = make_number(true); SkColor c = make_color(); SkDropShadowImageFilter::ShadowMode mode = make_shadow_mode(); sk_sp<SkImageFilter> fil = make_image_filter(); filter = SkDropShadowImageFilter::Make(dx, dy, sx, sy, c, mode, fil, nullptr); break; } case MORPHOLOGY: { int rx, ry; fuzz->nextRange(&rx, 0, kBitmapSize); fuzz->nextRange(&ry, 0, kBitmapSize); sk_sp<SkImageFilter> fil = make_image_filter(); if (make_bool()) { filter = SkDilateImageFilter::Make(rx, ry, fil); } else { filter = SkErodeImageFilter::Make(rx, ry, fil); } break; } case BITMAP: { sk_sp<SkImage> image(SkImage::MakeFromBitmap(make_bitmap())); if (make_bool()) { filter = SkImageSource::Make(std::move(image), make_rect(), make_rect(), kHigh_SkFilterQuality); } else { filter = SkImageSource::Make(std::move(image)); } break; } case DISPLACE: { SkDisplacementMapEffect::ChannelSelectorType x = make_channel_selector_type(); SkDisplacementMapEffect::ChannelSelectorType y = make_channel_selector_type(); SkScalar scale = make_number(false); sk_sp<SkImageFilter> filA = make_image_filter(false); sk_sp<SkImageFilter> filB = make_image_filter(); filter = SkDisplacementMapEffect::Make(x, y, scale, filA, filB); break; } case TILE: { SkRect src = make_rect(); SkRect dest = make_rect(); sk_sp<SkImageFilter> fil = make_image_filter(false); filter = SkTileImageFilter::Make(src, dest, fil); break; } case PICTURE: { SkRTreeFactory factory; SkPictureRecorder recorder; SkCanvas* recordingCanvas = recorder.beginRecording(SkIntToScalar(kBitmapSize), SkIntToScalar(kBitmapSize), &factory, 0); drawSomething(recordingCanvas); sk_sp<SkPicture> pict(recorder.finishRecordingAsPicture()); filter = SkPictureImageFilter::Make(pict, make_rect()); break; } case PAINT: { SkImageFilter::CropRect cropR(make_rect()); filter = SkPaintImageFilter::Make(make_paint(), &cropR); break; } default: break; } return filter; } static sk_sp<SkImageFilter> make_serialized_image_filter() { sk_sp<SkImageFilter> filter(make_image_filter(false)); sk_sp<SkData> data(SkValidatingSerializeFlattenable(filter.get())); const unsigned char* ptr = static_cast<const unsigned char*>(data->data()); size_t len = data->size(); #ifdef SK_ADD_RANDOM_BIT_FLIPS unsigned char* p = const_cast<unsigned char*>(ptr); for (size_t i = 0; i < len; ++i, ++p) { uint8_t j; fuzz->nextRange(&j, 1, 250); if (j == 1) { // 0.4% of the time, flip a bit or byte uint8_t k; fuzz->nextRange(&k, 1, 10); if (k == 1) { // Then 10% of the time, change a whole byte uint8_t s; fuzz->nextRange(&s, 0, 2); switch(s) { case 0: *p ^= 0xFF; // Flip entire byte break; case 1: *p = 0xFF; // Set all bits to 1 break; case 2: *p = 0x00; // Set all bits to 0 break; } } else { uint8_t s; fuzz->nextRange(&s, 0, 7); *p ^= (1 << 7); } } } #endif // SK_ADD_RANDOM_BIT_FLIPS return SkValidatingDeserializeImageFilter(ptr, len); } static void drawClippedBitmap(SkCanvas* canvas, int x, int y, const SkPaint& paint) { canvas->save(); canvas->clipRect(SkRect::MakeXYWH(SkIntToScalar(x), SkIntToScalar(y), SkIntToScalar(kBitmapSize), SkIntToScalar(kBitmapSize))); canvas->drawBitmap(make_bitmap(), SkIntToScalar(x), SkIntToScalar(y), &paint); canvas->restore(); } DEF_FUZZ(SerializedImageFilter, f) { fuzz = f; SkPaint paint; paint.setImageFilter(make_serialized_image_filter()); SkBitmap bitmap; SkCanvas canvas(bitmap); drawClippedBitmap(&canvas, 0, 0, paint); }