| /* |
| * Copyright 2020 Google, LLC |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkDeferredDisplayList.h" |
| #include "include/core/SkDeferredDisplayListRecorder.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkSurface.h" |
| #include "include/core/SkSurfaceCharacterization.h" |
| #include "include/gpu/GrDirectContext.h" |
| #include "include/private/GrTypesPriv.h" |
| #include "src/gpu/GrShaderCaps.h" |
| #include "tools/gpu/GrContextFactory.h" |
| |
| #include "fuzz/Fuzz.h" |
| |
| #include <tuple> |
| |
| /** |
| * The fuzzer aims to fuzz the use of SkDeferredDisplayList. It mainly consists of |
| * three parts. |
| * 1. In create_surface_characterization, (make_characterization) Create SkSurfaceCharacterization |
| * by using GrDirectContext of kGL_ContextType as it can be applied on all platform, and |
| * (make_surface) create a GPU backend surface of the same GrDirectContext |
| * 2. (make_ddl) Create SkDeferredDisplayListRecorder from the SkSurfaceCharacterization, and test |
| * the recoder's corresponding canvas. |
| * 3. (make_ddl, draw_ddl) Create SkDeferredDisplayList from the SkDeferredDisplayRecorder and draw |
| * the ddl on a GPU backend surface. |
| */ |
| |
| static constexpr int kMaxWidth = 64; |
| static constexpr int kMaxHeight = 64; |
| static constexpr int kSampleCount = 1; |
| |
| static SkSurfaceProps gen_fuzzed_surface_props(Fuzz* fuzz) { |
| SkPixelGeometry pixel; |
| fuzz->nextEnum(&pixel, kBGR_V_SkPixelGeometry); |
| return SkSurfaceProps(0x0, pixel); |
| } |
| |
| static SkPaint gen_fuzzed_skpaint(Fuzz* fuzz) { |
| float R, G, B, Alpha; |
| fuzz->nextRange(&R, -1, 2); |
| fuzz->nextRange(&G, -1, 2); |
| fuzz->nextRange(&B, -1, 2); |
| fuzz->nextRange(&Alpha, 0, 1); |
| SkColor4f color = {R, G, B, Alpha}; |
| return SkPaint(color); |
| } |
| |
| static SkImageInfo gen_fuzzed_imageinfo(Fuzz* fuzz, SkColorType surfaceType) { |
| int width, height; |
| fuzz->nextRange(&width, 1, kMaxWidth); |
| fuzz->nextRange(&height, 1, kMaxHeight); |
| SkAlphaType alphaType; |
| fuzz->nextEnum(&alphaType, SkAlphaType::kLastEnum_SkAlphaType); |
| skcms_TransferFunction skcmsFn; |
| uint8_t skcms; |
| fuzz->nextRange(&skcms, 0, 5); |
| switch (skcms) { |
| case 0: { |
| skcmsFn = SkNamedTransferFn::kSRGB; |
| break; |
| } |
| case 1: { |
| skcmsFn = SkNamedTransferFn::k2Dot2; |
| break; |
| } |
| case 2: { |
| skcmsFn = SkNamedTransferFn::kHLG; |
| break; |
| } |
| case 3: { |
| skcmsFn = SkNamedTransferFn::kLinear; |
| break; |
| } |
| case 4: { |
| skcmsFn = SkNamedTransferFn::kPQ; |
| break; |
| } |
| case 5: { |
| skcmsFn = SkNamedTransferFn::kRec2020; |
| break; |
| } |
| default: |
| SkASSERT(false); |
| break; |
| } |
| skcms_Matrix3x3 skcmsMat; |
| fuzz->nextRange(&skcms, 0, 4); |
| switch (skcms) { |
| case 0: { |
| skcmsMat = SkNamedGamut::kAdobeRGB; |
| break; |
| } |
| case 1: { |
| skcmsMat = SkNamedGamut::kDisplayP3; |
| break; |
| } |
| case 2: { |
| skcmsMat = SkNamedGamut::kRec2020; |
| break; |
| } |
| case 3: { |
| skcmsMat = SkNamedGamut::kSRGB; |
| break; |
| } |
| case 4: { |
| skcmsMat = SkNamedGamut::kXYZ; |
| break; |
| } |
| default: |
| SkASSERT(false); |
| break; |
| } |
| return SkImageInfo::Make(width, height, surfaceType, alphaType, |
| SkColorSpace::MakeRGB(skcmsFn, skcmsMat)); |
| } |
| |
| static SkSurfaceCharacterization make_characterization(Fuzz* fuzz, GrDirectContext* dContext, |
| SkImageInfo& ii, SkColorType surfaceType, |
| GrSurfaceOrigin origin) { |
| if (!dContext->colorTypeSupportedAsSurface(surfaceType)) { |
| SkDebugf("Couldn't create backend texture in the backend %s", |
| GrBackendApiToStr(dContext->backend())); |
| return {}; |
| } |
| |
| GrBackendFormat backendFormat = dContext->defaultBackendFormat(surfaceType, |
| GrRenderable::kYes); |
| if (!backendFormat.isValid()) { |
| SkDebugf("Color Type is not supported in the backend %s", |
| GrBackendApiToStr(dContext->backend())); |
| return {}; |
| } |
| GrProtected protect = GrProtected::kNo; |
| #ifdef SK_VULKAN |
| fuzz->nextEnum(&protect, GrProtected::kYes); |
| #endif |
| SkSurfaceCharacterization c; |
| size_t maxResourceBytes = dContext->getResourceCacheLimit(); |
| c = dContext->threadSafeProxy()->createCharacterization( |
| maxResourceBytes, ii, backendFormat, kSampleCount, |
| origin, gen_fuzzed_surface_props(fuzz), true, |
| false, true, protect); |
| if (!c.isValid()) { |
| SkDebugf("Could not create Characterization in the backend %s", |
| GrBackendApiToStr(dContext->backend())); |
| return {}; |
| } |
| return c; |
| } |
| |
| static sk_sp<SkDeferredDisplayList> make_ddl(Fuzz* fuzz, GrDirectContext* dContext, |
| const SkSurfaceCharacterization& c) { |
| SkDeferredDisplayListRecorder r(c); |
| SkCanvas* canvas = r.getCanvas(); |
| if (!canvas) { |
| SkDebugf("Could not create canvas for backend %s", GrBackendApiToStr(dContext->backend())); |
| return nullptr; |
| } |
| // For now we only draw a rect into the DDL. This will be scaled up to draw more varied content. |
| SkRect tile; |
| fuzz->next(&tile); |
| canvas->drawRect(tile, gen_fuzzed_skpaint(fuzz)); |
| return r.detach(); |
| } |
| |
| static sk_sp<SkSurface> make_surface(Fuzz* fuzz, GrDirectContext* dContext, const SkImageInfo& ii, |
| GrSurfaceOrigin origin) { |
| SkBudgeted budgeted; |
| fuzz->nextEnum(&budgeted, SkBudgeted::kYes); |
| SkSurfaceProps surfaceProps = gen_fuzzed_surface_props(fuzz); |
| auto surface = SkSurface::MakeRenderTarget(dContext, budgeted, ii, kSampleCount, origin, |
| &surfaceProps); |
| return surface; |
| } |
| |
| static bool draw_ddl(sk_sp<SkSurface> surface, sk_sp<SkDeferredDisplayList> ddl) { |
| return surface->draw(std::move(ddl)); |
| } |
| |
| using SurfaceAndChar = std::tuple<sk_sp<SkSurface>, SkSurfaceCharacterization>; |
| static SurfaceAndChar create_surface_and_characterization(Fuzz* fuzz, GrDirectContext* dContext, |
| SkColorType surfaceType, |
| GrSurfaceOrigin origin) { |
| SkImageInfo ii = gen_fuzzed_imageinfo(fuzz, surfaceType); |
| SkSurfaceCharacterization c = make_characterization(fuzz, dContext, ii, surfaceType, origin); |
| if (!c.isValid()) { |
| return {}; |
| } |
| |
| auto surface = make_surface(fuzz, dContext, ii, origin); |
| if (!surface) { |
| return {}; |
| } |
| return {surface, c}; |
| } |
| |
| DEF_FUZZ(CreateDDL, fuzz) { |
| SkColorType surfaceType; |
| GrSurfaceOrigin origin; |
| fuzz->nextEnum(&surfaceType, SkColorType::kLastEnum_SkColorType); |
| fuzz->nextEnum(&origin, GrSurfaceOrigin::kTopLeft_GrSurfaceOrigin); |
| |
| sk_gpu_test::GrContextFactory factory; |
| auto ctxInfo = factory.getContextInfo(sk_gpu_test::GrContextFactory::kGL_ContextType); |
| |
| GrDirectContext* dContext = ctxInfo.directContext(); |
| if (!dContext) { |
| SkDebugf("Context creation failed"); |
| return; |
| } |
| |
| auto[surface, c] = create_surface_and_characterization(fuzz, dContext, surfaceType, origin); |
| if (!surface || !c.isValid()) { |
| return; |
| } |
| |
| sk_sp<SkDeferredDisplayList> ddl = make_ddl(fuzz, dContext, c); |
| if (!ddl) { |
| SkDebugf("Could not create ddl %s", GrBackendApiToStr(dContext->backend())); |
| return; |
| } |
| if (!draw_ddl(std::move(surface), std::move(ddl))) { |
| SkDebugf("Could not draw ddl in the backend"); |
| } |
| return; |
| } |