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