| /* |
| * Copyright 2017 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| // This is a GPU-backend specific test. |
| #include "include/android/SkImageAndroid.h" |
| #include "include/core/SkAlphaType.h" |
| #include "include/core/SkBitmap.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkColor.h" |
| #include "include/core/SkColorType.h" |
| #include "include/core/SkImage.h" |
| #include "include/core/SkImageInfo.h" |
| #include "include/core/SkMatrix.h" |
| #include "include/core/SkRefCnt.h" |
| #include "include/core/SkSamplingOptions.h" |
| #include "include/core/SkSurface.h" |
| #include "include/core/SkTileMode.h" |
| #include "include/core/SkTypes.h" |
| #include "include/gpu/GpuTypes.h" |
| #include "include/gpu/GrDirectContext.h" |
| #include "include/gpu/ganesh/SkSurfaceGanesh.h" |
| #include "include/gpu/mock/GrMockTypes.h" |
| #include "src/gpu/ganesh/GrFragmentProcessor.h" // IWYU pragma: keep |
| #include "src/gpu/ganesh/SkGr.h" |
| #include "src/gpu/ganesh/image/GrImageUtils.h" |
| #include "tests/CtsEnforcement.h" |
| #include "tests/Test.h" |
| #include "tools/gpu/FenceSync.h" |
| |
| #include <string> |
| |
| class GrRecordingContext; |
| struct GrContextOptions; |
| |
| using namespace sk_gpu_test; |
| |
| static bool surface_is_expected_color(SkSurface* surf, const SkImageInfo& ii, SkColor color) { |
| SkBitmap bm; |
| bm.allocPixels(ii); |
| |
| surf->readPixels(bm, 0, 0); |
| |
| for (int y = 0; y < bm.height(); ++y) { |
| for (int x = 0; x < bm.width(); ++x) { |
| if (bm.getColor(x, y) != color) { |
| return false; |
| } |
| } |
| } |
| |
| return true; |
| } |
| |
| static void basic_test(skiatest::Reporter* reporter, GrRecordingContext* rContext) { |
| skiatest::ReporterContext subtest(reporter, "basic_test"); |
| const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType); |
| |
| SkBitmap bm; |
| bm.allocPixels(ii); |
| |
| SkCanvas bmCanvas(bm); |
| bmCanvas.clear(SK_ColorRED); |
| |
| // We start off with the raster image being all red. |
| sk_sp<SkImage> img = SkImages::PinnableRasterFromBitmap(bm); |
| REPORTER_ASSERT(reporter, img, "PinnableImageFromBitmap returned null"); |
| |
| sk_sp<SkSurface> gpuSurface = SkSurfaces::RenderTarget(rContext, skgpu::Budgeted::kYes, ii); |
| SkCanvas* canvas = gpuSurface->getCanvas(); |
| |
| // w/o pinning - the gpu draw always reflects the current state of the underlying bitmap |
| { |
| canvas->drawImage(img, 0, 0); |
| REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorRED)); |
| |
| bmCanvas.clear(SK_ColorGREEN); |
| |
| canvas->drawImage(img, 0, 0); |
| REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN)); |
| } |
| |
| // w/ pinning - the gpu draw is stuck at the pinned state |
| { |
| bool ok = skgpu::ganesh::PinAsTexture(rContext, img.get()); // pin at blue |
| REPORTER_ASSERT(reporter, ok, "PinAsTexture did not succeed"); |
| |
| canvas->drawImage(img, 0, 0); |
| REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN)); |
| |
| bmCanvas.clear(SK_ColorBLUE); |
| |
| canvas->drawImage(img, 0, 0); |
| REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorGREEN)); |
| |
| skgpu::ganesh::UnpinTexture(rContext, img.get()); |
| } |
| |
| // once unpinned local changes will be picked up |
| { |
| canvas->drawImage(img, 0, 0); |
| REPORTER_ASSERT(reporter, surface_is_expected_color(gpuSurface.get(), ii, SK_ColorBLUE)); |
| } |
| } |
| |
| // Deleting the context while there are still pinned images shouldn't result in a crash. |
| static void cleanup_test(skiatest::Reporter* reporter) { |
| skiatest::ReporterContext subtest(reporter, "cleanup_test"); |
| const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType); |
| |
| SkBitmap bm; |
| bm.allocPixels(ii); |
| |
| SkCanvas bmCanvas(bm); |
| bmCanvas.clear(SK_ColorRED); |
| |
| GrMockOptions options; |
| sk_sp<GrDirectContext> mockContext = GrDirectContext::MakeMock(&options); |
| |
| for (int i = 0; i < GrContextFactory::kContextTypeCnt; ++i) { |
| GrContextFactory::ContextType ctxType = (GrContextFactory::ContextType) i; |
| |
| { |
| sk_sp<SkImage> img; |
| GrDirectContext* dContext = nullptr; |
| |
| { |
| GrContextFactory testFactory; |
| ContextInfo info = testFactory.getContextInfo(ctxType); |
| dContext = info.directContext(); |
| if (!dContext) { |
| continue; |
| } |
| |
| img = SkImages::PinnableRasterFromBitmap(bm); |
| if (!skgpu::ganesh::PinAsTexture(dContext, img.get())) { |
| continue; |
| } |
| // Pinning on a second context should be blocked. |
| REPORTER_ASSERT(reporter, !skgpu::ganesh::PinAsTexture(mockContext.get(), |
| img.get())); |
| } |
| |
| // The context used to pin the image is gone at this point! |
| // "context" isn't technically used in this call but it can't be null! |
| // We don't really want to support this use case but it currently happens. |
| skgpu::ganesh::UnpinTexture(dContext, img.get()); |
| } |
| } |
| } |
| |
| DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(PinnedImageTest, |
| reporter, |
| ctxInfo, |
| CtsEnforcement::kApiLevel_T) { |
| |
| basic_test(reporter, ctxInfo.directContext()); |
| cleanup_test(reporter); |
| } |
| |
| DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(PinnedImageTest_AsGaneshView, |
| reporter, |
| ctxInfo, |
| CtsEnforcement::kApiLevel_T) { |
| GrRecordingContext* rContext = ctxInfo.directContext(); |
| const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType); |
| |
| SkBitmap bm; |
| bm.allocPixels(ii); |
| |
| SkCanvas bmCanvas(bm); |
| bmCanvas.clear(SK_ColorMAGENTA); // arbitrary color |
| |
| sk_sp<SkImage> img = SkImages::PinnableRasterFromBitmap(bm); |
| REPORTER_ASSERT(reporter, img, "PinnableImageFromBitmap returned null"); |
| |
| { |
| skiatest::ReporterContext subtest(reporter, "cached path"); |
| auto [view, colortype] = skgpu::ganesh::AsView(rContext, img, skgpu::Mipmapped::kNo, |
| GrImageTexGenPolicy::kDraw); |
| REPORTER_ASSERT(reporter, view, "AsView returned falsey view"); |
| } |
| |
| { |
| skiatest::ReporterContext subtest(reporter, "unncached path"); |
| auto [view, colortype] = skgpu::ganesh::AsView(rContext, img, skgpu::Mipmapped::kNo, |
| GrImageTexGenPolicy::kNew_Uncached_Unbudgeted); |
| REPORTER_ASSERT(reporter, view, "AsView returned falsey view"); |
| } |
| } |
| |
| DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(PinnedImageTest_AsFragmentProcessor, |
| reporter, |
| ctxInfo, |
| CtsEnforcement::kApiLevel_T) { |
| GrRecordingContext* rContext = ctxInfo.directContext(); |
| const SkImageInfo ii = SkImageInfo::Make(64, 64, kN32_SkColorType, kPremul_SkAlphaType); |
| |
| SkBitmap bm; |
| bm.allocPixels(ii); |
| |
| SkCanvas bmCanvas(bm); |
| bmCanvas.clear(SK_ColorMAGENTA); // arbitrary color |
| |
| sk_sp<SkImage> img = SkImages::PinnableRasterFromBitmap(bm); |
| REPORTER_ASSERT(reporter, img, "PinnableImageFromBitmap returned null"); |
| |
| SkTileMode tm[2] = {SkTileMode::kClamp, SkTileMode::kClamp}; |
| |
| auto fp = skgpu::ganesh::AsFragmentProcessor( |
| rContext, img.get(), SkSamplingOptions({1/3, 1/3}), tm, |
| SkMatrix::I(), nullptr, nullptr); |
| REPORTER_ASSERT(reporter, fp, "AsFragmentProcessor returned falsey processor"); |
| } |