/* * Copyright 2021 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/SkAlphaType.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/SkRect.h" #include "include/core/SkRefCnt.h" #include "include/core/SkSamplingOptions.h" #include "include/core/SkSurface.h" #include "include/core/SkTypes.h" #include "include/gpu/GpuTypes.h" #include "include/gpu/ganesh/GrDirectContext.h" #include "include/gpu/ganesh/GrTypes.h" #include "include/private/chromium/GrDeferredDisplayListRecorder.h" #include "include/private/chromium/GrSurfaceCharacterization.h" #include "include/private/gpu/ganesh/GrTypesPriv.h" #include "src/core/SkAutoPixmapStorage.h" #include "src/core/SkColorData.h" #include "src/gpu/SkBackingFit.h" #include "src/gpu/ganesh/GrCanvas.h" #include "src/gpu/ganesh/GrDirectContextPriv.h" #include "src/gpu/ganesh/GrDrawingManager.h" #include "src/gpu/ganesh/GrImageInfo.h" #include "src/gpu/ganesh/GrRenderTargetProxy.h" #include "src/gpu/ganesh/GrRenderTask.h" #include "src/gpu/ganesh/GrSamplerState.h" #include "src/gpu/ganesh/GrSurfaceProxy.h" #include "src/gpu/ganesh/GrSurfaceProxyView.h" #include "src/gpu/ganesh/GrTextureProxy.h" #include "src/gpu/ganesh/SurfaceContext.h" #include "src/gpu/ganesh/SurfaceFillContext.h" #include "tests/CtsEnforcement.h" #include "tests/Test.h" #include "tests/TestUtils.h" #include "tools/gpu/BackendSurfaceFactory.h" #include "tools/gpu/ProxyUtils.h" #include #include struct GrContextOptions; DEF_GANESH_TEST_FOR_ALL_CONTEXTS(WrappedSurfaceCopyOnWrite, reporter, ctxInfo, CtsEnforcement::kApiLevel_T) { using namespace skgpu; GrDirectContext* dContext = ctxInfo.directContext(); Protected isProtected = Protected(dContext->priv().caps()->supportsProtectedContent()); auto makeDirectBackendSurface = [&]() { auto info = SkImageInfo::Make({10, 10}, kRGBA_8888_SkColorType, kPremul_SkAlphaType); return sk_gpu_test::MakeBackendTextureSurface(dContext, info, kTopLeft_GrSurfaceOrigin, /* sampleCnt= */ 1, Mipmapped::kNo, isProtected); }; auto imageProxyID = [&](const sk_sp& img) { return sk_gpu_test::GetTextureImageProxy(img.get(), dContext)->uniqueID(); }; auto surfaceProxyID = [&](const sk_sp& surf) { GrRenderTargetProxy* rtp = skgpu::ganesh::TopDeviceTargetProxy(surf->getCanvas()); return rtp->uniqueID(); }; sk_sp surf = makeDirectBackendSurface(); surf->getCanvas()->clear(SkColor4f{1, 0, 0, 1}); sk_sp img = surf->makeImageSnapshot(); // Initially they share REPORTER_ASSERT(reporter, surfaceProxyID(surf) == imageProxyID(img)); // Using the image on the direct context shouldn't affect sharing. sk_sp surf2 = makeDirectBackendSurface(); surf2->getCanvas()->drawImage(img, 0, 0); REPORTER_ASSERT(reporter, surfaceProxyID(surf) == imageProxyID(img)); // Modifying the original surface should trigger using the copy proxy. surf->getCanvas()->clear({0, 0, 1, 1}); REPORTER_ASSERT(reporter, surfaceProxyID(surf) != imageProxyID(img)); // Image caching on surface should mean another snapshot gives us the same image. GrSurfaceProxy::UniqueID imageID = imageProxyID(img); img = surf->makeImageSnapshot(); REPORTER_ASSERT(reporter, imageProxyID(img) != imageID); GrSurfaceCharacterization characterization; REPORTER_ASSERT(reporter, surf->characterize(&characterization)); GrDeferredDisplayListRecorder recorder(characterization); // Using an image from a direct context on a recording context should trigger using the copy. surf = makeDirectBackendSurface(); img = surf->makeImageSnapshot(); REPORTER_ASSERT(reporter, surfaceProxyID(surf) == imageProxyID(img)); recorder.getCanvas()->drawImage(img, 0, 0); REPORTER_ASSERT(reporter, surfaceProxyID(surf) != imageProxyID(img)); // Same as above but if the surface goes out of scope first we keep using the original surf = makeDirectBackendSurface(); img = surf->makeImageSnapshot(); GrSurfaceProxy::UniqueID surfID = surfaceProxyID(surf); REPORTER_ASSERT(reporter, surfaceProxyID(surf) == imageProxyID(img)); surf.reset(); recorder.getCanvas()->drawImage(img, 0, 0); REPORTER_ASSERT(reporter, surfID == imageProxyID(img)); } // Make sure GrCopyRenderTasks's skip actually skips the copy. DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(SkipCopyTaskTest, reporter, ctxInfo, CtsEnforcement::kNever) { GrDirectContext* dContext = ctxInfo.directContext(); GrImageInfo info(GrColorType::kRGBA_8888, kPremul_SkAlphaType, /*color space*/ nullptr, 10, 10); auto dstSC = CreateSurfaceContext(dContext, info, SkBackingFit::kExact, kBottomLeft_GrSurfaceOrigin, GrRenderable::kYes); dstSC->asFillContext()->clear(SkPMColor4f{1, 0, 0, 1}); auto srcSC = CreateSurfaceContext(dContext, info, SkBackingFit::kExact, kBottomLeft_GrSurfaceOrigin, GrRenderable::kYes); srcSC->asFillContext()->clear(SkPMColor4f{0, 0, 1, 1}); sk_sp task = dContext->priv().drawingManager()->newCopyRenderTask(dstSC->asSurfaceProxyRef(), SkIRect::MakeWH(10, 10), srcSC->asSurfaceProxyRef(), SkIRect::MakeWH(10, 10), GrSamplerState::Filter::kNearest, kTopLeft_GrSurfaceOrigin); if (!task) { ERRORF(reporter, "Couldn't make a copy task."); return; } task->makeSkippable(); SkAutoPixmapStorage pixels; pixels.alloc(SkImageInfo::Make({10, 10}, kRGBA_8888_SkColorType, kPremul_SkAlphaType)); dstSC->readPixels(dContext, pixels, {0, 0}); float kTol[4] = {}; std::function errorReporter( [&](int x, int y, const float diffs[4]) { ERRORF(reporter, "Expected {1, 0, 0, 1}. diff {%f, %f, %f, %f}", diffs[0], diffs[1], diffs[2], diffs[3]); }); CheckSolidPixels(SkColor4f{1, 0, 0, 1}, pixels, kTol, errorReporter); } #if defined(SK_GANESH) // Make sure OpsTask are skippable DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(SkipOpsTaskTest, reporter, ctxInfo, CtsEnforcement::kNever) { GrDirectContext* dContext = ctxInfo.directContext(); GrImageInfo ii(GrColorType::kRGBA_8888, kPremul_SkAlphaType, /*color space*/ nullptr, 10, 10); auto dst = dContext->priv().makeSFC(ii, /*label=*/{}, SkBackingFit::kExact); dst->clear(SkPMColor4f{1, 0, 0, 1}); dContext->flush(); dst->clear(SkPMColor4f{0, 0, 1, 1}); sk_sp task = dst->refRenderTask(); // GrDrawingManager maintains an "active ops task" and doesn't like having it closed behind // its back. temp exists just to replace dst's ops task as the active one. auto temp = dContext->priv().makeSFC(ii, /*label=*/{}, SkBackingFit::kExact); temp->clear(SkPMColor4f{0, 0, 0, 0}); GrSurfaceProxyView readView = dst->readSurfaceView(); task->makeSkippable(); SkAutoPixmapStorage pixels; pixels.alloc(SkImageInfo::Make({10, 10}, kRGBA_8888_SkColorType, kPremul_SkAlphaType)); dst->readPixels(dContext, pixels, {0, 0}); float kTol[4] = {}; std::function errorReporter( [&](int x, int y, const float diffs[4]) { ERRORF(reporter, "Expected {1, 0, 0, 1}. diff {%f, %f, %f, %f}", diffs[0], diffs[1], diffs[2], diffs[3]); }); CheckSolidPixels(SkColor4f{1, 0, 0, 1}, pixels, kTol, errorReporter); } #endif // defined(SK_GANESH)