/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file */ #include "include/core/SkAlphaType.h" #include "include/core/SkBitmap.h" #include "include/core/SkCanvas.h" #include "include/core/SkColor.h" #include "include/core/SkColorSpace.h" #include "include/core/SkColorType.h" #include "include/core/SkImage.h" #include "include/core/SkImageInfo.h" #include "include/core/SkPaint.h" #include "include/core/SkRect.h" #include "include/core/SkRefCnt.h" #include "include/core/SkScalar.h" #include "include/core/SkSize.h" #include "include/core/SkSurface.h" #include "include/core/SkSurfaceProps.h" #include "include/core/SkTypes.h" #include "include/gpu/GpuTypes.h" #include "include/gpu/ganesh/GrDirectContext.h" #include "include/gpu/ganesh/SkSurfaceGanesh.h" #include "src/core/SkSpecialImage.h" #include "src/gpu/ganesh/GrColorInfo.h" // IWYU pragma: keep #include "src/gpu/ganesh/GrSurfaceProxyView.h" #include "src/gpu/ganesh/SkGr.h" #include "src/gpu/ganesh/image/SkSpecialImage_Ganesh.h" #include "tests/CtsEnforcement.h" #include "tests/Test.h" #include class GrRecordingContext; struct GrContextOptions; // This test creates backing resources exactly sized to [kFullSize x kFullSize]. // It then wraps them in an SkSpecialImage with only the center (red) region being active. // It then draws the SkSpecialImage to a full sized (all blue) canvas and checks that none // of the inactive (green) region leaked out. static const int kSmallerSize = 10; static const int kPad = 3; static const int kFullSize = kSmallerSize + 2 * kPad; // Create a bitmap with red in the center and green around it static SkBitmap create_bm() { SkImageInfo ii = SkImageInfo::Make(kFullSize, kFullSize, kRGBA_8888_SkColorType, kPremul_SkAlphaType); SkBitmap bm; bm.allocPixels(ii); SkCanvas temp(bm); temp.clear(SK_ColorGREEN); SkPaint p; p.setColor(SK_ColorRED); p.setAntiAlias(false); temp.drawRect(SkRect::MakeXYWH(SkIntToScalar(kPad), SkIntToScalar(kPad), SkIntToScalar(kSmallerSize), SkIntToScalar(kSmallerSize)), p); bm.setImmutable(); return bm; } // Basic test of the SkSpecialImage public API (e.g., peekTexture, peekPixels & draw) static void test_image(const sk_sp& img, skiatest::Reporter* reporter, GrRecordingContext* rContext, bool isGPUBacked) { const SkIRect subset = img->subset(); REPORTER_ASSERT(reporter, kPad == subset.left()); REPORTER_ASSERT(reporter, kPad == subset.top()); REPORTER_ASSERT(reporter, kSmallerSize == subset.width()); REPORTER_ASSERT(reporter, kSmallerSize == subset.height()); //-------------- // Test that isTextureBacked reports the correct backing type REPORTER_ASSERT(reporter, isGPUBacked == img->isGaneshBacked()); REPORTER_ASSERT(reporter, !img->isGraphiteBacked()); //-------------- // Test view - only succeeds if it's Ganesh backed if (rContext) { GrSurfaceProxyView view = SkSpecialImages::AsView(rContext, img); REPORTER_ASSERT(reporter, SkToBool(view.asTextureProxy()) == isGPUBacked); } //-------------- // Test AsBitmap - this only works for raster-backed special images if (!img->isGaneshBacked() && !img->isGraphiteBacked()) { SkBitmap bitmap; REPORTER_ASSERT(reporter, SkSpecialImages::AsBitmap(img.get(), &bitmap)); REPORTER_ASSERT(reporter, kSmallerSize == bitmap.width()); REPORTER_ASSERT(reporter, kSmallerSize == bitmap.height()); } else { SkBitmap bitmap; REPORTER_ASSERT(reporter, !SkSpecialImages::AsBitmap(img.get(), &bitmap)); } //-------------- // Test that draw restricts itself to the subset SkImageInfo imageInfo = SkImageInfo::Make(SkISize::Make(kFullSize, kFullSize), kN32_SkColorType, kPremul_SkAlphaType, sk_ref_sp(img->getColorSpace())); sk_sp surf = isGPUBacked ? SkSurfaces::RenderTarget(rContext, skgpu::Budgeted::kNo, imageInfo) : SkSurfaces::Raster(imageInfo, {}); SkCanvas* canvas = surf->getCanvas(); canvas->clear(SK_ColorBLUE); img->draw(canvas, SkIntToScalar(kPad), SkIntToScalar(kPad)); SkBitmap bm; bm.allocN32Pixels(kFullSize, kFullSize, false); bool result = canvas->readPixels(bm.info(), bm.getPixels(), bm.rowBytes(), 0, 0); SkASSERT_RELEASE(result); // Only the center (red) portion should've been drawn into the canvas REPORTER_ASSERT(reporter, SK_ColorBLUE == bm.getColor(kPad-1, kPad-1)); REPORTER_ASSERT(reporter, SK_ColorRED == bm.getColor(kPad, kPad)); REPORTER_ASSERT(reporter, SK_ColorRED == bm.getColor(kSmallerSize+kPad-1, kSmallerSize+kPad-1)); REPORTER_ASSERT(reporter, SK_ColorBLUE == bm.getColor(kSmallerSize+kPad, kSmallerSize+kPad)); } DEF_TEST(SpecialImage_Raster, reporter) { SkBitmap bm = create_bm(); sk_sp fullSImage(SkSpecialImages::MakeFromRaster( SkIRect::MakeWH(kFullSize, kFullSize), bm, SkSurfaceProps())); const SkIRect& subset = SkIRect::MakeXYWH(kPad, kPad, kSmallerSize, kSmallerSize); { sk_sp subSImg1( SkSpecialImages::MakeFromRaster(subset, bm, SkSurfaceProps())); test_image(subSImg1, reporter, nullptr, false); } { sk_sp subSImg2(fullSImage->makeSubset(subset)); test_image(subSImg2, reporter, nullptr, false); } } static void test_specialimage_image(skiatest::Reporter* reporter) { SkBitmap bm = create_bm(); sk_sp fullImage(bm.asImage()); sk_sp fullSImage(SkSpecialImages::MakeFromRaster( SkIRect::MakeWH(kFullSize, kFullSize), fullImage, SkSurfaceProps())); const SkIRect& subset = SkIRect::MakeXYWH(kPad, kPad, kSmallerSize, kSmallerSize); { sk_sp subSImg1( SkSpecialImages::MakeFromRaster(subset, fullImage, SkSurfaceProps())); test_image(subSImg1, reporter, nullptr, false); } { sk_sp subSImg2(fullSImage->makeSubset(subset)); test_image(subSImg2, reporter, nullptr, false); } } DEF_TEST(SpecialImage_Image_Legacy, reporter) { test_specialimage_image(reporter); } DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(SpecialImage_Gpu, reporter, ctxInfo, CtsEnforcement::kApiLevel_T) { auto context = ctxInfo.directContext(); SkBitmap bm = create_bm(); auto [view, ct] = GrMakeUncachedBitmapProxyView(context, bm); if (!view) { return; } sk_sp fullSImg = SkSpecialImages::MakeDeferredFromGpu(context, SkIRect::MakeWH(kFullSize, kFullSize), kNeedNewImageUniqueID_SpecialImage, view, {ct, kPremul_SkAlphaType, nullptr}, SkSurfaceProps()); const SkIRect& subset = SkIRect::MakeXYWH(kPad, kPad, kSmallerSize, kSmallerSize); { sk_sp subSImg1 = SkSpecialImages::MakeDeferredFromGpu(context, subset, kNeedNewImageUniqueID_SpecialImage, std::move(view), {ct, kPremul_SkAlphaType, nullptr}, SkSurfaceProps()); test_image(subSImg1, reporter, context, true); } { sk_sp subSImg2 = fullSImg->makeSubset(subset); test_image(subSImg2, reporter, context, true); } }