| /* |
| * Copyright 2017 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" |
| |
| // CORE |
| #include "SkCanvas.h" |
| #include "SkColorFilter.h" |
| #include "SkDebugCanvas.h" |
| #include "SkDocument.h" |
| #include "SkFontMgr.h" |
| #include "SkImageFilter.h" |
| #include "SkMaskFilter.h" |
| #include "SkNullCanvas.h" |
| #include "SkPathEffect.h" |
| #include "SkPictureRecorder.h" |
| #include "SkRegion.h" |
| #include "SkSurface.h" |
| #include "SkTypeface.h" |
| |
| // EFFECTS |
| #include "SkColorMatrixFilter.h" |
| #include "SkGaussianEdgeShader.h" |
| #include "SkGradientShader.h" |
| #include "SkHighContrastFilter.h" |
| #include "SkLumaColorFilter.h" |
| #include "SkPerlinNoiseShader.h" |
| #include "SkTableColorFilter.h" |
| |
| // SRC |
| #include "SkUtils.h" |
| |
| // MISC |
| |
| #include <iostream> |
| |
| // TODO: |
| // SkCanvas::drawTextBlob |
| // SkCanvas::drawTextRSXform |
| // SkImageFilter |
| // SkMaskFilter |
| // SkPathEffect |
| |
| template <typename T, void (SkPaint::*S)(T)> |
| inline void fuzz_input(Fuzz* fuzz, SkPaint* paint) { |
| T value; |
| fuzz->next(&value); |
| (paint->*S)(value); |
| } |
| |
| template <typename T, void (SkPaint::*S)(T)> |
| inline void fuzz_enum_input(Fuzz* fuzz, SkPaint* paint, T rmin, T rmax) { |
| using U = skstd::underlying_type_t<T>; |
| U value; |
| fuzz->nextRange(&value, (U)rmin, (U)rmax); |
| (paint->*S)((T)value); |
| } |
| |
| // be careful: `foo(make_bool(f), make_bool(f))` is undefined. |
| static bool make_bool(Fuzz* fuzz) { |
| bool b; |
| fuzz->next(&b); |
| return b; |
| } |
| |
| // We don't always want to test NaNs. |
| static void fuzz_nice_float(Fuzz* fuzz, float* f) { |
| fuzz->next(f); |
| if (*f != *f || ::fabs(*f) > 1.0e35f) { |
| *f = 0.0f; |
| } |
| } |
| |
| template <typename... Args> |
| void fuzz_nice_float(Fuzz* fuzz, float* f, Args... rest) { |
| fuzz_nice_float(fuzz, f); |
| fuzz_nice_float(fuzz, rest...); |
| } |
| |
| static void fuzz_path(Fuzz* fuzz, SkPath* path, int maxOps) { |
| if (maxOps < 2) { |
| maxOps = 2; |
| } |
| uint8_t fillType; |
| fuzz->nextRange(&fillType, 0, (uint8_t)SkPath::kInverseEvenOdd_FillType); |
| path->setFillType((SkPath::FillType)fillType); |
| uint8_t numOps; |
| fuzz->nextRange(&numOps, 2, maxOps); |
| for (uint8_t i = 0; i < numOps; ++i) { |
| uint8_t op; |
| fuzz->nextRange(&op, 0, 6); |
| SkScalar a, b, c, d, e, f; |
| switch (op) { |
| case 0: |
| fuzz_nice_float(fuzz, &a, &b); |
| path->moveTo(a, b); |
| break; |
| case 1: |
| fuzz_nice_float(fuzz, &a, &b); |
| path->lineTo(a, b); |
| break; |
| case 2: |
| fuzz_nice_float(fuzz, &a, &b, &c, &d); |
| path->quadTo(a, b, c, d); |
| break; |
| case 3: |
| fuzz_nice_float(fuzz, &a, &b, &c, &d, &e); |
| path->conicTo(a, b, c, d, e); |
| break; |
| case 4: |
| fuzz_nice_float(fuzz, &a, &b, &c, &d, &e, &f); |
| path->cubicTo(a, b, c, d, e, f); |
| break; |
| case 5: |
| fuzz_nice_float(fuzz, &a, &b, &c, &d, &e); |
| path->arcTo(a, b, c, d, e); |
| break; |
| case 6: |
| path->close(); |
| break; |
| default: |
| break; |
| } |
| } |
| } |
| |
| static void fuzz_region(Fuzz* fuzz, SkRegion* region) { |
| uint8_t N; |
| fuzz->nextRange(&N, 0, 10); |
| for (uint8_t i = 0; i < N; ++i) { |
| SkIRect r; |
| uint8_t op; |
| fuzz->next(&r); |
| r.sort(); |
| fuzz->nextRange(&op, 0, (uint8_t)SkRegion::kLastOp); |
| if (!region->op(r, (SkRegion::Op)op)) { |
| return; |
| } |
| } |
| } |
| |
| template <> |
| inline void Fuzz::next(SkShader::TileMode* m) { |
| using U = skstd::underlying_type_t<SkShader::TileMode>; |
| this->nextRange((U*)m, (U)0, (U)(SkShader::kTileModeCount - 1)); |
| } |
| |
| template <> |
| inline void Fuzz::next(SkMatrix* m) { |
| constexpr int kArrayLength = 9; |
| SkScalar buffer[kArrayLength]; |
| int matrixType; |
| this->nextRange(&matrixType, 0, 4); |
| switch (matrixType) { |
| case 0: // identity |
| *m = SkMatrix::I(); |
| return; |
| case 1: // translate |
| this->nextRange(&buffer[0], -4000.0f, 4000.0f); |
| this->nextRange(&buffer[1], -4000.0f, 4000.0f); |
| *m = SkMatrix::MakeTrans(buffer[0], buffer[1]); |
| return; |
| case 2: // translate + scale |
| this->nextRange(&buffer[0], -400.0f, 400.0f); |
| this->nextRange(&buffer[1], -400.0f, 400.0f); |
| this->nextRange(&buffer[2], -4000.0f, 4000.0f); |
| this->nextRange(&buffer[3], -4000.0f, 4000.0f); |
| *m = SkMatrix::MakeScale(buffer[0], buffer[1]); |
| m->postTranslate(buffer[2], buffer[3]); |
| return; |
| case 3: // affine |
| this->nextN(buffer, 6); |
| m->setAffine(buffer); |
| return; |
| case 4: // perspective |
| this->nextN(buffer, kArrayLength); |
| m->set9(buffer); |
| return; |
| default: |
| return; |
| } |
| } |
| |
| template <> |
| inline void Fuzz::next(SkRRect* rr) { |
| SkRect r; |
| SkVector radii[4]; |
| this->next(&r); |
| r.sort(); |
| for (SkVector& vec : radii) { |
| this->nextRange(&vec.fX, 0.0f, 1.0f); |
| vec.fX *= 0.5f * r.width(); |
| this->nextRange(&vec.fY, 0.0f, 1.0f); |
| vec.fY *= 0.5f * r.height(); |
| } |
| rr->setRectRadii(r, radii); |
| } |
| |
| template <> |
| inline void Fuzz::next(SkBlendMode* mode) { |
| using U = skstd::underlying_type_t<SkBlendMode>; |
| this->nextRange((U*)mode, (U)0, (U)SkBlendMode::kLastMode); |
| } |
| |
| sk_sp<SkImage> MakeFuzzImage(Fuzz*); |
| |
| SkBitmap MakeFuzzBitmap(Fuzz*); |
| |
| static sk_sp<SkPicture> make_picture(Fuzz*, int depth); |
| |
| sk_sp<SkColorFilter> MakeColorFilter(Fuzz* fuzz, int depth = 3) { |
| if (depth <= 0) { |
| return nullptr; |
| } |
| int colorFilterType; |
| fuzz->nextRange(&colorFilterType, 0, 8); |
| switch (colorFilterType) { |
| case 0: |
| return nullptr; |
| case 1: { |
| SkColor color; |
| SkBlendMode mode; |
| fuzz->next(&color, &mode); |
| return SkColorFilter::MakeModeFilter(color, mode); |
| } |
| case 2: { |
| sk_sp<SkColorFilter> outer = MakeColorFilter(fuzz, depth - 1); |
| sk_sp<SkColorFilter> inner = MakeColorFilter(fuzz, depth - 1); |
| return SkColorFilter::MakeComposeFilter(std::move(outer), std::move(inner)); |
| } |
| case 3: { |
| SkScalar array[20]; |
| fuzz->nextN(array, SK_ARRAY_COUNT(array)); |
| return SkColorFilter::MakeMatrixFilterRowMajor255(array); |
| } |
| case 4: { |
| SkColor mul, add; |
| fuzz->next(&mul, &add); |
| return SkColorMatrixFilter::MakeLightingFilter(mul, add); |
| } |
| case 5: { |
| bool grayscale; |
| int invertStyle; |
| float contrast; |
| fuzz->next(&grayscale); |
| fuzz->nextRange(&invertStyle, 0, 2); |
| fuzz->nextRange(&contrast, -1.0f, 1.0f); |
| return SkHighContrastFilter::Make(SkHighContrastConfig( |
| grayscale, SkHighContrastConfig::InvertStyle(invertStyle), contrast)); |
| } |
| case 6: |
| return SkLumaColorFilter::Make(); |
| case 7: { |
| uint8_t table[256]; |
| fuzz->nextN(table, SK_ARRAY_COUNT(table)); |
| return SkTableColorFilter::Make(table); |
| } |
| case 8: { |
| uint8_t tableA[256]; |
| uint8_t tableR[256]; |
| uint8_t tableG[256]; |
| uint8_t tableB[256]; |
| fuzz->nextN(tableA, SK_ARRAY_COUNT(tableA)); |
| fuzz->nextN(tableR, SK_ARRAY_COUNT(tableR)); |
| fuzz->nextN(tableG, SK_ARRAY_COUNT(tableG)); |
| fuzz->nextN(tableB, SK_ARRAY_COUNT(tableB)); |
| return SkTableColorFilter::MakeARGB(tableA, tableR, tableG, tableB); |
| } |
| } |
| return nullptr; |
| } |
| |
| void make_pos(Fuzz* fuzz, SkScalar* pos, int colorCount) { |
| SkScalar totalPos = 0; |
| for (int i = 0; i < colorCount; ++i) { |
| fuzz->nextRange(&pos[i], 1.0f, 1024.0f); |
| totalPos += pos[i]; |
| } |
| totalPos = 1.0f / totalPos; |
| for (int i = 0; i < colorCount; ++i) { |
| pos[i] *= totalPos; |
| } |
| // SkASSERT(fabs(pos[colorCount - 1] - 1.0f) < 0.00001f); |
| pos[colorCount - 1] = 1.0f; |
| } |
| |
| sk_sp<SkShader> MakeFuzzShader(Fuzz* fuzz, int depth) { |
| sk_sp<SkShader> shader1(nullptr), shader2(nullptr); |
| sk_sp<SkColorFilter> colorFilter(nullptr); |
| SkBitmap bitmap; |
| sk_sp<SkImage> img; |
| SkShader::TileMode tmX, tmY; |
| bool useMatrix; |
| SkColor color; |
| SkMatrix matrix; |
| SkBlendMode blendMode; |
| int shaderType; |
| if (depth <= 0) { |
| return nullptr; |
| } |
| fuzz->nextRange(&shaderType, 0, 14); |
| switch (shaderType) { |
| case 0: |
| return nullptr; |
| case 1: |
| return SkShader::MakeEmptyShader(); |
| case 2: |
| fuzz->next(&color); |
| return SkShader::MakeColorShader(color); |
| case 3: |
| img = MakeFuzzImage(fuzz); |
| fuzz->next(&tmX, &tmY, &useMatrix); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| return img->makeShader(tmX, tmY, useMatrix ? &matrix : nullptr); |
| case 4: |
| bitmap = MakeFuzzBitmap(fuzz); |
| fuzz->next(&tmX, &tmY, &useMatrix); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| return SkShader::MakeBitmapShader(bitmap, tmX, tmY, useMatrix ? &matrix : nullptr); |
| case 5: |
| shader1 = MakeFuzzShader(fuzz, depth - 1); // limit recursion. |
| fuzz->next(&matrix); |
| return shader1 ? shader1->makeWithLocalMatrix(matrix) : nullptr; |
| case 6: |
| shader1 = MakeFuzzShader(fuzz, depth - 1); // limit recursion. |
| colorFilter = MakeColorFilter(fuzz); |
| return shader1 ? shader1->makeWithColorFilter(std::move(colorFilter)) : nullptr; |
| case 7: |
| shader1 = MakeFuzzShader(fuzz, depth - 1); // limit recursion. |
| shader2 = MakeFuzzShader(fuzz, depth - 1); |
| fuzz->next(&blendMode); |
| return SkShader::MakeComposeShader(std::move(shader1), std::move(shader2), blendMode); |
| case 8: { |
| auto pic = make_picture(fuzz, depth); |
| bool useTile; |
| SkRect tile; |
| fuzz->next(&tmX, &tmY, &useMatrix, &useTile); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| if (useTile) { |
| fuzz->next(&tile); |
| } |
| return SkShader::MakePictureShader(std::move(pic), tmX, tmY, |
| useMatrix ? &matrix : nullptr, |
| useTile ? &tile : nullptr); |
| } |
| // EFFECTS: |
| case 9: |
| return SkGaussianEdgeShader::Make(); |
| case 10: { |
| constexpr int kMaxColors = 12; |
| SkPoint pts[2]; |
| SkColor colors[kMaxColors]; |
| SkScalar pos[kMaxColors]; |
| int colorCount; |
| bool usePos; |
| fuzz->nextN(pts, 2); |
| fuzz->nextRange(&colorCount, 2, kMaxColors); |
| fuzz->nextN(colors, colorCount); |
| fuzz->next(&tmX, &useMatrix, &usePos); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| if (usePos) { |
| make_pos(fuzz, pos, colorCount); |
| } |
| return SkGradientShader::MakeLinear(pts, colors, usePos ? pos : nullptr, colorCount, |
| tmX, 0, useMatrix ? &matrix : nullptr); |
| } |
| case 11: { |
| constexpr int kMaxColors = 12; |
| SkPoint center; |
| SkScalar radius; |
| int colorCount; |
| bool usePos; |
| SkColor colors[kMaxColors]; |
| SkScalar pos[kMaxColors]; |
| fuzz->next(&tmX, &useMatrix, &usePos, ¢er, &radius); |
| fuzz->nextRange(&colorCount, 2, kMaxColors); |
| fuzz->nextN(colors, colorCount); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| if (usePos) { |
| make_pos(fuzz, pos, colorCount); |
| } |
| return SkGradientShader::MakeRadial(center, radius, colors, usePos ? pos : nullptr, |
| colorCount, tmX, 0, useMatrix ? &matrix : nullptr); |
| } |
| case 12: { |
| constexpr int kMaxColors = 12; |
| SkPoint start, end; |
| SkScalar startRadius, endRadius; |
| int colorCount; |
| bool usePos; |
| SkColor colors[kMaxColors]; |
| SkScalar pos[kMaxColors]; |
| fuzz->next(&tmX, &useMatrix, &usePos, &startRadius, &endRadius, &start, &end); |
| fuzz->nextRange(&colorCount, 2, kMaxColors); |
| fuzz->nextN(colors, colorCount); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| if (usePos) { |
| make_pos(fuzz, pos, colorCount); |
| } |
| return SkGradientShader::MakeTwoPointConical(start, startRadius, end, endRadius, colors, |
| usePos ? pos : nullptr, colorCount, tmX, 0, |
| useMatrix ? &matrix : nullptr); |
| } |
| case 13: { |
| constexpr int kMaxColors = 12; |
| SkScalar cx, cy; |
| int colorCount; |
| bool usePos; |
| SkColor colors[kMaxColors]; |
| SkScalar pos[kMaxColors]; |
| fuzz->next(&cx, &cy, &useMatrix, &usePos); |
| fuzz->nextRange(&colorCount, 2, kMaxColors); |
| fuzz->nextN(colors, colorCount); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| if (usePos) { |
| make_pos(fuzz, pos, colorCount); |
| } |
| return SkGradientShader::MakeSweep(cx, cy, colors, usePos ? pos : nullptr, colorCount, |
| 0, useMatrix ? &matrix : nullptr); |
| } |
| case 14: { |
| SkScalar baseFrequencyX, baseFrequencyY, seed; |
| int numOctaves; |
| SkISize tileSize; |
| bool useTileSize, turbulence; |
| fuzz->next(&baseFrequencyX, &baseFrequencyY, &seed, &useTileSize, &turbulence); |
| if (useTileSize) { |
| fuzz->next(&tileSize); |
| } |
| fuzz->nextRange(&numOctaves, 2, 7); |
| if (turbulence) { |
| return SkPerlinNoiseShader::MakeTurbulence(baseFrequencyX, baseFrequencyY, |
| numOctaves, seed, |
| useTileSize ? &tileSize : nullptr); |
| } else { |
| return SkPerlinNoiseShader::MakeFractalNoise(baseFrequencyX, baseFrequencyY, |
| numOctaves, seed, |
| useTileSize ? &tileSize : nullptr); |
| } |
| } |
| default: |
| break; |
| } |
| return nullptr; |
| } |
| |
| sk_sp<SkPathEffect> MakeFuzzPathEffect(Fuzz* fuzz) { return nullptr; /*TODO*/ } |
| |
| sk_sp<SkMaskFilter> MakeFuzzMaskFilter(Fuzz* fuzz) { return nullptr; /*TODO*/ } |
| |
| sk_sp<SkTypeface> MakeFuzzTypeface(Fuzz* fuzz) { |
| if (make_bool(fuzz)) { |
| return nullptr; |
| } |
| auto fontMugger = SkFontMgr::RefDefault(); |
| SkASSERT(fontMugger); |
| int familyCount = fontMugger->countFamilies(); |
| int i, j; |
| fuzz->nextRange(&i, 0, familyCount - 1); |
| sk_sp<SkFontStyleSet> family(fontMugger->createStyleSet(i)); |
| int styleCount = family->count(); |
| fuzz->nextRange(&j, 0, styleCount - 1); |
| return sk_sp<SkTypeface>(family->createTypeface(j)); |
| } |
| |
| sk_sp<SkImageFilter> MakeFuzzImageFilter(Fuzz* fuzz) { return nullptr; /*TODO*/ } |
| |
| sk_sp<SkImage> MakeFuzzImage(Fuzz* fuzz) { |
| int w, h; |
| fuzz->nextRange(&w, 1, 1024); |
| fuzz->nextRange(&h, 1, 1024); |
| SkAutoTMalloc<SkPMColor> data(w * h); |
| SkPixmap pixmap(SkImageInfo::MakeN32Premul(w, h), data.get(), w * sizeof(SkPMColor)); |
| int n = w * h; |
| for (int i = 0; i < n; ++i) { |
| SkColor c; |
| fuzz->next(&c); |
| data[i] = SkPreMultiplyColor(c); |
| } |
| (void)data.release(); |
| return SkImage::MakeFromRaster(pixmap, [](const void* p, void*) { sk_free((void*)p); }, |
| nullptr); |
| } |
| |
| SkBitmap MakeFuzzBitmap(Fuzz* fuzz) { |
| SkBitmap bitmap; |
| int w, h; |
| fuzz->nextRange(&w, 1, 1024); |
| fuzz->nextRange(&h, 1, 1024); |
| bitmap.allocN32Pixels(w, h); |
| SkAutoLockPixels autoLockPixels(bitmap); |
| for (int y = 0; y < h; ++y) { |
| for (int x = 0; x < w; ++x) { |
| SkColor c; |
| fuzz->next(&c); |
| *bitmap.getAddr32(x, y) = SkPreMultiplyColor(c); |
| } |
| } |
| return bitmap; |
| } |
| |
| void FuzzPaint(Fuzz* fuzz, SkPaint* paint, int depth) { |
| if (!fuzz || !paint || depth <= 0) { |
| return; |
| } |
| |
| fuzz_input<bool, &SkPaint::setAntiAlias>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setDither>(fuzz, paint); |
| fuzz_input<SkColor, &SkPaint::setColor>(fuzz, paint); |
| fuzz_enum_input<SkBlendMode, &SkPaint::setBlendMode>(fuzz, paint, (SkBlendMode)0, |
| SkBlendMode::kLastMode); |
| fuzz_enum_input<SkPaint::Hinting, &SkPaint::setHinting>(fuzz, paint, SkPaint::kNo_Hinting, |
| SkPaint::kFull_Hinting); |
| fuzz_enum_input<SkFilterQuality, &SkPaint::setFilterQuality>( |
| fuzz, paint, SkFilterQuality::kNone_SkFilterQuality, |
| SkFilterQuality::kLast_SkFilterQuality); |
| fuzz_enum_input<SkPaint::Style, &SkPaint::setStyle>(fuzz, paint, SkPaint::kFill_Style, |
| SkPaint::kStrokeAndFill_Style); |
| paint->setShader(MakeFuzzShader(fuzz, depth)); |
| paint->setPathEffect(MakeFuzzPathEffect(fuzz)); |
| paint->setMaskFilter(MakeFuzzMaskFilter(fuzz)); |
| paint->setImageFilter(MakeFuzzImageFilter(fuzz)); |
| paint->setColorFilter(MakeColorFilter(fuzz)); |
| |
| if (paint->getStyle() != SkPaint::kFill_Style) { |
| fuzz_input<SkScalar, &SkPaint::setStrokeWidth>(fuzz, paint); |
| fuzz_input<SkScalar, &SkPaint::setStrokeMiter>(fuzz, paint); |
| fuzz_enum_input<SkPaint::Cap, &SkPaint::setStrokeCap>(fuzz, paint, SkPaint::kButt_Cap, |
| SkPaint::kLast_Cap); |
| fuzz_enum_input<SkPaint::Join, &SkPaint::setStrokeJoin>(fuzz, paint, SkPaint::kMiter_Join, |
| SkPaint::kLast_Join); |
| } |
| } |
| |
| void FuzzPaintText(Fuzz* fuzz, SkPaint* paint) { |
| paint->setTypeface(MakeFuzzTypeface(fuzz)); |
| fuzz_input<SkScalar, &SkPaint::setTextSize>(fuzz, paint); |
| fuzz_input<SkScalar, &SkPaint::setTextScaleX>(fuzz, paint); |
| fuzz_input<SkScalar, &SkPaint::setTextSkewX>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setLinearText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setSubpixelText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setLCDRenderText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setEmbeddedBitmapText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setAutohinted>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setVerticalText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setFakeBoldText>(fuzz, paint); |
| fuzz_input<bool, &SkPaint::setDevKernText>(fuzz, paint); |
| fuzz_enum_input<SkPaint::Align, &SkPaint::setTextAlign>(fuzz, paint, SkPaint::kLeft_Align, |
| SkPaint::kRight_Align); |
| fuzz_enum_input<SkPaint::TextEncoding, &SkPaint::setTextEncoding>( |
| fuzz, paint, SkPaint::kUTF8_TextEncoding, SkPaint::kGlyphID_TextEncoding); |
| } |
| |
| SkTDArray<uint8_t> fuzz_text(Fuzz* fuzz, const SkPaint& paint) { |
| SkTDArray<uint8_t> array; |
| if (SkPaint::kGlyphID_TextEncoding == paint.getTextEncoding()) { |
| int glyphRange = paint.getTypeface() ? paint.getTypeface()->countGlyphs() |
| : SkTypeface::MakeDefault()->countGlyphs(); |
| constexpr int kMaxGlyphCount = 20; |
| int glyphCount; |
| fuzz->nextRange(&glyphCount, 0, kMaxGlyphCount); |
| SkGlyphID* glyphs = (SkGlyphID*)array.append(glyphCount * sizeof(SkGlyphID)); |
| for (int i = 0; i < glyphCount; ++i) { |
| fuzz->nextRange(&glyphs[i], 0, glyphRange - 1); |
| } |
| return array; |
| } |
| static const SkUnichar ranges[][2] = { |
| {0x0020, 0x007F}, |
| {0x00A1, 0x0250}, |
| {0x0400, 0x0500}, |
| }; |
| int32_t count = 0; |
| for (size_t i = 0; i < SK_ARRAY_COUNT(ranges); ++i) { |
| count += (ranges[i][1] - ranges[i][0]); |
| } |
| constexpr int kMaxLength = 30; |
| SkUnichar buffer[kMaxLength]; |
| int length; |
| fuzz->nextRange(&length, 1, kMaxLength); |
| for (int j = 0; j < length; ++j) { |
| int32_t value; |
| fuzz->nextRange(&value, 0, count - 1); |
| for (size_t i = 0; i < SK_ARRAY_COUNT(ranges); ++i) { |
| if (value + ranges[i][0] < ranges[i][1]) { |
| buffer[j] = value + ranges[i][0]; |
| break; |
| } else { |
| value -= (ranges[i][1] - ranges[i][0]); |
| } |
| } |
| } |
| switch (paint.getTextEncoding()) { |
| case SkPaint::kUTF8_TextEncoding: { |
| size_t utf8len = 0; |
| for (int j = 0; j < length; ++j) { |
| utf8len += SkUTF8_FromUnichar(buffer[j], nullptr); |
| } |
| char* ptr = (char*)array.append(utf8len); |
| for (int j = 0; j < length; ++j) { |
| ptr += SkUTF8_FromUnichar(buffer[j], ptr); |
| } |
| } break; |
| case SkPaint::kUTF16_TextEncoding: { |
| size_t utf16len = 0; |
| for (int j = 0; j < length; ++j) { |
| utf16len += SkUTF16_FromUnichar(buffer[j]); |
| } |
| uint16_t* ptr = (uint16_t*)array.append(utf16len * sizeof(uint16_t)); |
| for (int j = 0; j < length; ++j) { |
| ptr += SkUTF16_FromUnichar(buffer[j], ptr); |
| } |
| } break; |
| case SkPaint::kUTF32_TextEncoding: |
| memcpy(array.append(length * sizeof(SkUnichar)), buffer, length * sizeof(SkUnichar)); |
| break; |
| default: |
| SkASSERT(false); |
| } |
| return array; |
| } |
| |
| void fuzz_canvas(Fuzz* fuzz, SkCanvas* canvas, int depth = 4) { |
| if (!fuzz || !canvas || depth <= 0) { |
| return; |
| } |
| SkAutoCanvasRestore autoCanvasRestore(canvas, false); |
| unsigned N; |
| fuzz->nextRange(&N, 0, 2000); |
| for (unsigned i = 0; i < N; ++i) { |
| if (fuzz->exhausted()) { |
| return; |
| } |
| SkPaint paint; |
| SkMatrix matrix; |
| unsigned drawCommand; |
| fuzz->nextRange(&drawCommand, 0, 54); |
| switch (drawCommand) { |
| case 0: |
| canvas->flush(); |
| break; |
| case 1: |
| canvas->save(); |
| break; |
| case 2: { |
| SkRect bounds; |
| fuzz->next(&bounds); |
| FuzzPaint(fuzz, &paint, depth); |
| canvas->saveLayer(&bounds, &paint); |
| break; |
| } |
| case 3: { |
| SkRect bounds; |
| fuzz->next(&bounds); |
| canvas->saveLayer(&bounds, nullptr); |
| break; |
| } |
| case 4: |
| FuzzPaint(fuzz, &paint, depth); |
| canvas->saveLayer(nullptr, &paint); |
| break; |
| case 5: |
| canvas->saveLayer(nullptr, nullptr); |
| break; |
| case 6: { |
| uint8_t alpha; |
| fuzz->next(&alpha); |
| canvas->saveLayerAlpha(nullptr, (U8CPU)alpha); |
| break; |
| } |
| case 7: { |
| SkRect bounds; |
| uint8_t alpha; |
| fuzz->next(&bounds, &alpha); |
| canvas->saveLayerAlpha(&bounds, (U8CPU)alpha); |
| break; |
| } |
| case 8: { |
| SkCanvas::SaveLayerRec saveLayerRec; |
| SkRect bounds; |
| if (make_bool(fuzz)) { |
| fuzz->next(&bounds); |
| saveLayerRec.fBounds = &bounds; |
| } |
| if (make_bool(fuzz)) { |
| FuzzPaint(fuzz, &paint, depth); |
| saveLayerRec.fPaint = &paint; |
| } |
| sk_sp<SkImageFilter> imageFilter; |
| if (make_bool(fuzz)) { |
| imageFilter = MakeFuzzImageFilter(fuzz); |
| saveLayerRec.fBackdrop = imageFilter.get(); |
| } |
| if (make_bool(fuzz)) { |
| saveLayerRec.fSaveLayerFlags |= SkCanvas::kIsOpaque_SaveLayerFlag; |
| } |
| if (make_bool(fuzz)) { |
| saveLayerRec.fSaveLayerFlags |= SkCanvas::kPreserveLCDText_SaveLayerFlag; |
| } |
| canvas->saveLayer(saveLayerRec); |
| break; |
| } |
| case 9: |
| canvas->restore(); |
| break; |
| case 10: { |
| int saveCount; |
| fuzz->next(&saveCount); |
| canvas->restoreToCount(saveCount); |
| break; |
| } |
| case 11: { |
| SkScalar x, y; |
| fuzz->next(&x, &y); |
| canvas->translate(x, y); |
| break; |
| } |
| case 12: { |
| SkScalar x, y; |
| fuzz->next(&x, &y); |
| canvas->scale(x, y); |
| break; |
| } |
| case 13: { |
| SkScalar v; |
| fuzz->next(&v); |
| canvas->rotate(v); |
| break; |
| } |
| case 14: { |
| SkScalar x, y, v; |
| fuzz->next(&x, &y, &v); |
| canvas->rotate(v, x, y); |
| break; |
| } |
| case 15: { |
| SkScalar x, y; |
| fuzz->next(&x, &y); |
| canvas->skew(x, y); |
| break; |
| } |
| case 16: { |
| SkMatrix mat; |
| fuzz->next(&mat); |
| canvas->concat(mat); |
| break; |
| } |
| case 17: { |
| SkMatrix mat; |
| fuzz->next(&mat); |
| canvas->setMatrix(mat); |
| break; |
| } |
| case 18: |
| canvas->resetMatrix(); |
| break; |
| case 19: { |
| SkRect r; |
| int op; |
| bool doAntiAlias; |
| fuzz->next(&r, &doAntiAlias); |
| fuzz->nextRange(&op, 0, 1); |
| r.sort(); |
| canvas->clipRect(r, (SkClipOp)op, doAntiAlias); |
| break; |
| } |
| case 20: { |
| SkRRect rr; |
| int op; |
| bool doAntiAlias; |
| fuzz->next(&rr); |
| fuzz->next(&doAntiAlias); |
| fuzz->nextRange(&op, 0, 1); |
| canvas->clipRRect(rr, (SkClipOp)op, doAntiAlias); |
| break; |
| } |
| case 21: { |
| SkPath path; |
| fuzz_path(fuzz, &path, 30); |
| int op; |
| bool doAntiAlias; |
| fuzz->next(&doAntiAlias); |
| fuzz->nextRange(&op, 0, 1); |
| canvas->clipPath(path, (SkClipOp)op, doAntiAlias); |
| break; |
| } |
| case 22: { |
| SkRegion region; |
| fuzz_region(fuzz, ®ion); |
| int op; |
| fuzz->nextRange(&op, 0, 1); |
| canvas->clipRegion(region, (SkClipOp)op); |
| break; |
| } |
| case 23: |
| FuzzPaint(fuzz, &paint, depth); |
| canvas->drawPaint(paint); |
| break; |
| case 24: { |
| FuzzPaint(fuzz, &paint, depth); |
| uint8_t pointMode; |
| fuzz->nextRange(&pointMode, 0, 3); |
| size_t count; |
| constexpr int kMaxCount = 30; |
| fuzz->nextRange(&count, 0, kMaxCount); |
| SkPoint pts[kMaxCount]; |
| fuzz->nextN(pts, count); |
| canvas->drawPoints((SkCanvas::PointMode)pointMode, count, pts, paint); |
| break; |
| } |
| case 25: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRect r; |
| fuzz->next(&r); |
| canvas->drawRect(r, paint); |
| break; |
| } |
| case 26: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRegion region; |
| fuzz_region(fuzz, ®ion); |
| canvas->drawRegion(region, paint); |
| break; |
| } |
| case 27: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRect r; |
| fuzz->next(&r); |
| canvas->drawOval(r, paint); |
| break; |
| } |
| case 29: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRRect rr; |
| fuzz->next(&rr); |
| canvas->drawRRect(rr, paint); |
| break; |
| } |
| case 30: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRRect orr, irr; |
| fuzz->next(&orr); |
| fuzz->next(&irr); |
| if (orr.getBounds().contains(irr.getBounds())) { |
| canvas->drawDRRect(orr, irr, paint); |
| } |
| break; |
| } |
| case 31: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkRect r; |
| SkScalar start, sweep; |
| bool useCenter; |
| fuzz->next(&r, &start, &sweep, &useCenter); |
| canvas->drawArc(r, start, sweep, useCenter, paint); |
| break; |
| } |
| case 32: { |
| SkPath path; |
| fuzz_path(fuzz, &path, 60); |
| canvas->drawPath(path, paint); |
| break; |
| } |
| case 33: { |
| sk_sp<SkImage> img = MakeFuzzImage(fuzz); |
| SkScalar left, top; |
| bool usePaint; |
| fuzz->next(&left, &top, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| canvas->drawImage(img.get(), left, top, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 34: { |
| auto img = MakeFuzzImage(fuzz); |
| SkRect src, dst; |
| bool usePaint; |
| fuzz->next(&src, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawImageRect(img, src, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 35: { |
| auto img = MakeFuzzImage(fuzz); |
| SkIRect src; |
| SkRect dst; |
| bool usePaint; |
| fuzz->next(&src, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawImageRect(img, src, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 36: { |
| bool usePaint; |
| auto img = MakeFuzzImage(fuzz); |
| SkRect dst; |
| fuzz->next(&dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawImageRect(img, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 37: { |
| auto img = MakeFuzzImage(fuzz); |
| SkIRect center; |
| SkRect dst; |
| bool usePaint; |
| fuzz->next(¢er, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| canvas->drawImageNine(img, center, dst, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 38: { |
| SkBitmap bitmap = MakeFuzzBitmap(fuzz); |
| SkScalar left, top; |
| bool usePaint; |
| fuzz->next(&left, &top, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| canvas->drawBitmap(bitmap, left, top, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 39: { |
| SkBitmap bitmap = MakeFuzzBitmap(fuzz); |
| SkRect src, dst; |
| bool usePaint; |
| fuzz->next(&src, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawBitmapRect(bitmap, src, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 40: { |
| SkBitmap img = MakeFuzzBitmap(fuzz); |
| SkIRect src; |
| SkRect dst; |
| bool usePaint; |
| fuzz->next(&src, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawBitmapRect(img, src, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 41: { |
| SkBitmap img = MakeFuzzBitmap(fuzz); |
| SkRect dst; |
| bool usePaint; |
| fuzz->next(&dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| SkCanvas::SrcRectConstraint constraint = |
| make_bool(fuzz) ? SkCanvas::kStrict_SrcRectConstraint |
| : SkCanvas::kFast_SrcRectConstraint; |
| canvas->drawBitmapRect(img, dst, usePaint ? &paint : nullptr, constraint); |
| break; |
| } |
| case 42: { |
| SkBitmap img = MakeFuzzBitmap(fuzz); |
| SkIRect center; |
| SkRect dst; |
| bool usePaint; |
| fuzz->next(¢er, &dst, &usePaint); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| canvas->drawBitmapNine(img, center, dst, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 43: { |
| SkBitmap img = MakeFuzzBitmap(fuzz); |
| bool usePaint; |
| SkRect dst; |
| fuzz->next(&usePaint, &dst); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| constexpr int kMax = 6; |
| int xDivs[kMax], yDivs[kMax]; |
| SkCanvas::Lattice lattice{xDivs, yDivs, nullptr, 0, 0, nullptr}; |
| fuzz->nextRange(&lattice.fXCount, 2, kMax); |
| fuzz->nextRange(&lattice.fYCount, 2, kMax); |
| fuzz->nextN(xDivs, lattice.fXCount); |
| fuzz->nextN(yDivs, lattice.fYCount); |
| canvas->drawBitmapLattice(img, lattice, dst, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 44: { |
| auto img = MakeFuzzImage(fuzz); |
| bool usePaint; |
| SkRect dst; |
| fuzz->next(&usePaint, &dst); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| constexpr int kMax = 6; |
| int xDivs[kMax], yDivs[kMax]; |
| SkCanvas::Lattice lattice{xDivs, yDivs, nullptr, 0, 0, nullptr}; |
| fuzz->nextRange(&lattice.fXCount, 2, kMax); |
| fuzz->nextRange(&lattice.fYCount, 2, kMax); |
| fuzz->nextN(xDivs, lattice.fXCount); |
| fuzz->nextN(yDivs, lattice.fYCount); |
| canvas->drawImageLattice(img.get(), lattice, dst, usePaint ? &paint : nullptr); |
| break; |
| } |
| case 45: { |
| FuzzPaint(fuzz, &paint, depth); |
| FuzzPaintText(fuzz, &paint); |
| SkScalar x, y; |
| fuzz->next(&x, &y); |
| SkTDArray<uint8_t> text = fuzz_text(fuzz, paint); |
| canvas->drawText(text.begin(), SkToSizeT(text.count()), x, y, paint); |
| break; |
| } |
| case 46: { |
| FuzzPaint(fuzz, &paint, depth); |
| FuzzPaintText(fuzz, &paint); |
| SkTDArray<uint8_t> text = fuzz_text(fuzz, paint); |
| int glyphCount = paint.countText(text.begin(), SkToSizeT(text.count())); |
| if (glyphCount < 1) { |
| break; |
| } |
| SkAutoTMalloc<SkPoint> pos(glyphCount); |
| SkAutoTMalloc<SkScalar> widths(glyphCount); |
| paint.getTextWidths(text.begin(), SkToSizeT(text.count()), widths.get()); |
| pos[0] = {0, 0}; |
| for (int i = 1; i < glyphCount; ++i) { |
| float y; |
| fuzz->nextRange(&y, -0.5f * paint.getTextSize(), 0.5f * paint.getTextSize()); |
| pos[i] = {pos[i - 1].x() + widths[i - 1], y}; |
| } |
| canvas->drawPosText(text.begin(), SkToSizeT(text.count()), pos.get(), paint); |
| break; |
| } |
| case 47: { |
| FuzzPaint(fuzz, &paint, depth); |
| FuzzPaintText(fuzz, &paint); |
| SkTDArray<uint8_t> text = fuzz_text(fuzz, paint); |
| int glyphCount = paint.countText(text.begin(), SkToSizeT(text.count())); |
| SkAutoTMalloc<SkScalar> widths(glyphCount); |
| if (glyphCount < 1) { |
| break; |
| } |
| paint.getTextWidths(text.begin(), SkToSizeT(text.count()), widths.get()); |
| SkScalar x = widths[0]; |
| for (int i = 0; i < glyphCount; ++i) { |
| SkTSwap(x, widths[i]); |
| x += widths[i]; |
| SkScalar offset; |
| fuzz->nextRange(&offset, -0.125f * paint.getTextSize(), |
| 0.125f * paint.getTextSize()); |
| widths[i] += offset; |
| } |
| SkScalar y; |
| fuzz->next(&y); |
| canvas->drawPosTextH(text.begin(), SkToSizeT(text.count()), widths.get(), y, paint); |
| break; |
| } |
| case 48: { |
| FuzzPaint(fuzz, &paint, depth); |
| FuzzPaintText(fuzz, &paint); |
| SkTDArray<uint8_t> text = fuzz_text(fuzz, paint); |
| SkPath path; |
| fuzz_path(fuzz, &path, 20); |
| SkScalar hOffset, vOffset; |
| fuzz->next(&hOffset, &vOffset); |
| canvas->drawTextOnPathHV(text.begin(), SkToSizeT(text.count()), path, hOffset, |
| vOffset, paint); |
| break; |
| } |
| case 49: { |
| SkMatrix matrix; |
| bool useMatrix = make_bool(fuzz); |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| FuzzPaint(fuzz, &paint, depth); |
| FuzzPaintText(fuzz, &paint); |
| SkTDArray<uint8_t> text = fuzz_text(fuzz, paint); |
| SkPath path; |
| fuzz_path(fuzz, &path, 20); |
| canvas->drawTextOnPath(text.begin(), SkToSizeT(text.count()), path, |
| useMatrix ? &matrix : nullptr, paint); |
| break; |
| } |
| case 50: { |
| // canvas->drawTextRSXform(...); // TODO |
| break; |
| } |
| case 51: { |
| // canvas->drawTextBlob(...); // TODO |
| break; |
| } |
| case 52: { |
| bool usePaint, useMatrix; |
| fuzz->next(&usePaint, &useMatrix); |
| if (usePaint) { |
| FuzzPaint(fuzz, &paint, depth); |
| } |
| if (useMatrix) { |
| fuzz->next(&matrix); |
| } |
| auto pic = make_picture(fuzz, depth); |
| canvas->drawPicture(pic, useMatrix ? &matrix : nullptr, |
| usePaint ? &paint : nullptr); |
| break; |
| } |
| case 53: { |
| FuzzPaint(fuzz, &paint, depth); |
| SkCanvas::VertexMode vertexMode; |
| SkBlendMode mode; |
| uint8_t vm, bm; |
| fuzz->nextRange(&vm, 0, (uint8_t)SkCanvas::kTriangleFan_VertexMode); |
| fuzz->nextRange(&bm, 0, (uint8_t)SkBlendMode::kLastMode); |
| vertexMode = (SkCanvas::VertexMode)vm; |
| mode = (SkBlendMode)bm; |
| constexpr int kMaxCount = 100; |
| int vertexCount; |
| SkPoint vertices[kMaxCount]; |
| SkPoint texs[kMaxCount]; |
| SkColor colors[kMaxCount]; |
| fuzz->nextRange(&vertexCount, 3, kMaxCount); |
| fuzz->nextN(vertices, vertexCount); |
| bool useTexs, useColors; |
| fuzz->next(&useTexs, &useColors); |
| if (useTexs) { |
| fuzz->nextN(texs, vertexCount); |
| } |
| if (useColors) { |
| fuzz->nextN(colors, vertexCount); |
| } |
| int indexCount = 0; |
| uint16_t indices[kMaxCount * 2]; |
| if (make_bool(fuzz)) { |
| fuzz->nextRange(&indexCount, vertexCount, vertexCount + kMaxCount); |
| for (int i = 0; i < indexCount; ++i) { |
| fuzz->nextRange(&indices[i], 0, vertexCount - 1); |
| } |
| } |
| canvas->drawVertices(vertexMode, vertexCount, vertices, useTexs ? texs : nullptr, |
| useColors ? colors : nullptr, mode, |
| indexCount > 0 ? indices : nullptr, indexCount, paint); |
| break; |
| } |
| case 54: { |
| // canvas->drawVertices(...); |
| // TODO |
| break; |
| } |
| default: |
| break; |
| } |
| } |
| } |
| |
| static sk_sp<SkPicture> make_picture(Fuzz* fuzz, int depth) { |
| SkScalar w, h; |
| fuzz->next(&w, &h); |
| SkPictureRecorder pictureRecorder; |
| fuzz_canvas(fuzz, pictureRecorder.beginRecording(w, h), depth - 1); |
| return pictureRecorder.finishRecordingAsPicture(); |
| } |
| |
| DEF_FUZZ(NullCanvas, fuzz) { |
| fuzz_canvas(fuzz, SkMakeNullCanvas().get()); |
| } |
| |
| DEF_FUZZ(RasterN32Canvas, fuzz) { |
| fuzz_canvas(fuzz, SkMakeNullCanvas().get()); |
| auto surface = SkSurface::MakeRasterN32Premul(612, 792); |
| SkASSERT(surface && surface->getCanvas()); |
| fuzz_canvas(fuzz, surface->getCanvas()); |
| } |
| |
| DEF_FUZZ(PDFCanvas, fuzz) { |
| struct final : public SkWStream { |
| bool write(const void*, size_t n) override { fN += n; return true; } |
| size_t bytesWritten() const override { return fN; } |
| size_t fN = 0; |
| } stream; |
| auto doc = SkDocument::MakePDF(&stream); |
| fuzz_canvas(fuzz, doc->beginPage(612.0f, 792.0f)); |
| } |
| |
| // not a "real" thing to fuzz, used to debug errors found while fuzzing. |
| DEF_FUZZ(_DumpCanvas, fuzz) { |
| SkDebugCanvas debugCanvas(612, 792); |
| fuzz_canvas(fuzz, &debugCanvas); |
| std::unique_ptr<SkCanvas> nullCanvas = SkMakeNullCanvas(); |
| UrlDataManager dataManager(SkString("data")); |
| Json::Value json = debugCanvas.toJSON(dataManager, debugCanvas.getSize(), nullCanvas.get()); |
| Json::StyledStreamWriter(" ").write(std::cout, json); |
| } |