1cb93a386Sopenharmony_ci/* 2cb93a386Sopenharmony_ci * Copyright 2016 Google Inc. 3cb93a386Sopenharmony_ci * 4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be 5cb93a386Sopenharmony_ci * found in the LICENSE file. 6cb93a386Sopenharmony_ci */ 7cb93a386Sopenharmony_ci 8cb93a386Sopenharmony_ci#include "include/core/SkImageInfo.h" 9cb93a386Sopenharmony_ci#include "include/core/SkPoint.h" 10cb93a386Sopenharmony_ci#include "include/core/SkRect.h" 11cb93a386Sopenharmony_ci#include "include/core/SkRefCnt.h" 12cb93a386Sopenharmony_ci#include "include/core/SkTypes.h" 13cb93a386Sopenharmony_ci#include "include/gpu/GrDirectContext.h" 14cb93a386Sopenharmony_ci#include "include/gpu/GrTypes.h" 15cb93a386Sopenharmony_ci#include "include/private/GrTypesPriv.h" 16cb93a386Sopenharmony_ci#include "include/private/SkTemplates.h" 17cb93a386Sopenharmony_ci#include "src/core/SkOpts.h" 18cb93a386Sopenharmony_ci#include "src/gpu/GrCaps.h" 19cb93a386Sopenharmony_ci#include "src/gpu/GrDirectContextPriv.h" 20cb93a386Sopenharmony_ci#include "src/gpu/GrImageInfo.h" 21cb93a386Sopenharmony_ci#include "src/gpu/GrSurfaceProxy.h" 22cb93a386Sopenharmony_ci#include "src/gpu/GrTextureProxy.h" 23cb93a386Sopenharmony_ci#include "src/gpu/SkGr.h" 24cb93a386Sopenharmony_ci#include "src/gpu/SurfaceFillContext.h" 25cb93a386Sopenharmony_ci#include "tests/Test.h" 26cb93a386Sopenharmony_ci#include "tools/gpu/GrContextFactory.h" 27cb93a386Sopenharmony_ci#include "tools/gpu/ProxyUtils.h" 28cb93a386Sopenharmony_ci 29cb93a386Sopenharmony_ci#include <initializer_list> 30cb93a386Sopenharmony_ci#include <utility> 31cb93a386Sopenharmony_ci 32cb93a386Sopenharmony_ciDEF_GPUTEST_FOR_RENDERING_CONTEXTS(CopySurface, reporter, ctxInfo) { 33cb93a386Sopenharmony_ci auto dContext = ctxInfo.directContext(); 34cb93a386Sopenharmony_ci 35cb93a386Sopenharmony_ci static const int kW = 10; 36cb93a386Sopenharmony_ci static const int kH = 10; 37cb93a386Sopenharmony_ci static const size_t kRowBytes = sizeof(uint32_t) * kW; 38cb93a386Sopenharmony_ci 39cb93a386Sopenharmony_ci SkAutoTMalloc<uint32_t> srcPixels(kW * kH); 40cb93a386Sopenharmony_ci for (int i = 0; i < kW * kH; ++i) { 41cb93a386Sopenharmony_ci srcPixels.get()[i] = i; 42cb93a386Sopenharmony_ci } 43cb93a386Sopenharmony_ci 44cb93a386Sopenharmony_ci SkAutoTMalloc<uint32_t> dstPixels(kW * kH); 45cb93a386Sopenharmony_ci for (int i = 0; i < kW * kH; ++i) { 46cb93a386Sopenharmony_ci dstPixels.get()[i] = ~i; 47cb93a386Sopenharmony_ci } 48cb93a386Sopenharmony_ci 49cb93a386Sopenharmony_ci static const SkIRect kSrcRects[] { 50cb93a386Sopenharmony_ci { 0, 0, kW , kH }, 51cb93a386Sopenharmony_ci {-1, -1, kW+1, kH+1}, 52cb93a386Sopenharmony_ci { 1, 1, kW-1, kH-1}, 53cb93a386Sopenharmony_ci { 5, 5, 6 , 6 }, 54cb93a386Sopenharmony_ci }; 55cb93a386Sopenharmony_ci 56cb93a386Sopenharmony_ci static const SkIPoint kDstPoints[] { 57cb93a386Sopenharmony_ci { 0 , 0 }, 58cb93a386Sopenharmony_ci { 1 , 1 }, 59cb93a386Sopenharmony_ci { kW/2, kH/4}, 60cb93a386Sopenharmony_ci { kW-1, kH-1}, 61cb93a386Sopenharmony_ci { kW , kH }, 62cb93a386Sopenharmony_ci { kW+1, kH+2}, 63cb93a386Sopenharmony_ci {-1 , -1 }, 64cb93a386Sopenharmony_ci }; 65cb93a386Sopenharmony_ci 66cb93a386Sopenharmony_ci static const SkImageInfo kImageInfos[] { 67cb93a386Sopenharmony_ci SkImageInfo::Make(kW, kH, kRGBA_8888_SkColorType, kPremul_SkAlphaType), 68cb93a386Sopenharmony_ci SkImageInfo::Make(kW, kH, kBGRA_8888_SkColorType, kPremul_SkAlphaType) 69cb93a386Sopenharmony_ci }; 70cb93a386Sopenharmony_ci 71cb93a386Sopenharmony_ci SkAutoTMalloc<uint32_t> read(kW * kH); 72cb93a386Sopenharmony_ci 73cb93a386Sopenharmony_ci for (auto sOrigin : {kBottomLeft_GrSurfaceOrigin, kTopLeft_GrSurfaceOrigin}) { 74cb93a386Sopenharmony_ci for (auto dOrigin : {kBottomLeft_GrSurfaceOrigin, kTopLeft_GrSurfaceOrigin}) { 75cb93a386Sopenharmony_ci for (auto sRenderable : {GrRenderable::kYes, GrRenderable::kNo}) { 76cb93a386Sopenharmony_ci for (auto dRenderable : {GrRenderable::kYes, GrRenderable::kNo}) { 77cb93a386Sopenharmony_ci for (const SkIRect& srcRect : kSrcRects) { 78cb93a386Sopenharmony_ci for (const SkIPoint& dstPoint : kDstPoints) { 79cb93a386Sopenharmony_ci for (const SkImageInfo& ii: kImageInfos) { 80cb93a386Sopenharmony_ci GrCPixmap srcPM(ii, srcPixels.get(), kRowBytes); 81cb93a386Sopenharmony_ci GrPixmap dstPM(ii, dstPixels.get(), kRowBytes); 82cb93a386Sopenharmony_ci auto srcView = sk_gpu_test::MakeTextureProxyViewFromData( 83cb93a386Sopenharmony_ci dContext, sRenderable, sOrigin, srcPM); 84cb93a386Sopenharmony_ci auto dstView = sk_gpu_test::MakeTextureProxyViewFromData( 85cb93a386Sopenharmony_ci dContext, dRenderable, dOrigin, dstPM); 86cb93a386Sopenharmony_ci 87cb93a386Sopenharmony_ci // Should always work if the color type is RGBA, but may not work 88cb93a386Sopenharmony_ci // for BGRA 89cb93a386Sopenharmony_ci if (ii.colorType() == kRGBA_8888_SkColorType) { 90cb93a386Sopenharmony_ci if (!srcView || !dstView) { 91cb93a386Sopenharmony_ci ERRORF(reporter, 92cb93a386Sopenharmony_ci "Could not create surfaces for copy surface test."); 93cb93a386Sopenharmony_ci continue; 94cb93a386Sopenharmony_ci } 95cb93a386Sopenharmony_ci } else { 96cb93a386Sopenharmony_ci if (!dContext->defaultBackendFormat( 97cb93a386Sopenharmony_ci kBGRA_8888_SkColorType, GrRenderable::kNo).isValid()) { 98cb93a386Sopenharmony_ci continue; 99cb93a386Sopenharmony_ci } 100cb93a386Sopenharmony_ci if (!srcView || !dstView) { 101cb93a386Sopenharmony_ci ERRORF(reporter, 102cb93a386Sopenharmony_ci "Could not create surfaces for copy surface test."); 103cb93a386Sopenharmony_ci continue; 104cb93a386Sopenharmony_ci } 105cb93a386Sopenharmony_ci } 106cb93a386Sopenharmony_ci 107cb93a386Sopenharmony_ci auto dstContext = dContext->priv().makeSC(std::move(dstView), 108cb93a386Sopenharmony_ci ii.colorInfo()); 109cb93a386Sopenharmony_ci 110cb93a386Sopenharmony_ci bool result = false; 111cb93a386Sopenharmony_ci if (sOrigin == dOrigin) { 112cb93a386Sopenharmony_ci result = dstContext->testCopy(srcView.refProxy(), 113cb93a386Sopenharmony_ci srcRect, 114cb93a386Sopenharmony_ci dstPoint); 115cb93a386Sopenharmony_ci } else if (dRenderable == GrRenderable::kYes) { 116cb93a386Sopenharmony_ci SkASSERT(dstContext->asFillContext()); 117cb93a386Sopenharmony_ci result = dstContext->asFillContext()->blitTexture( 118cb93a386Sopenharmony_ci std::move(srcView), srcRect, dstPoint); 119cb93a386Sopenharmony_ci } 120cb93a386Sopenharmony_ci 121cb93a386Sopenharmony_ci bool expectedResult = true; 122cb93a386Sopenharmony_ci SkIPoint dstOffset = { dstPoint.fX - srcRect.fLeft, 123cb93a386Sopenharmony_ci dstPoint.fY - srcRect.fTop }; 124cb93a386Sopenharmony_ci SkIRect copiedDstRect = SkIRect::MakeXYWH(dstPoint.fX, 125cb93a386Sopenharmony_ci dstPoint.fY, 126cb93a386Sopenharmony_ci srcRect.width(), 127cb93a386Sopenharmony_ci srcRect.height()); 128cb93a386Sopenharmony_ci 129cb93a386Sopenharmony_ci SkIRect copiedSrcRect; 130cb93a386Sopenharmony_ci if (!copiedSrcRect.intersect(srcRect, SkIRect::MakeWH(kW, kH))) { 131cb93a386Sopenharmony_ci expectedResult = false; 132cb93a386Sopenharmony_ci } else { 133cb93a386Sopenharmony_ci // If the src rect was clipped, apply same clipping to each side 134cb93a386Sopenharmony_ci // of copied dst rect. 135cb93a386Sopenharmony_ci copiedDstRect.fLeft += copiedSrcRect.fLeft - srcRect.fLeft; 136cb93a386Sopenharmony_ci copiedDstRect.fTop += copiedSrcRect.fTop - srcRect.fTop; 137cb93a386Sopenharmony_ci copiedDstRect.fRight -= copiedSrcRect.fRight - srcRect.fRight; 138cb93a386Sopenharmony_ci copiedDstRect.fBottom -= copiedSrcRect.fBottom - 139cb93a386Sopenharmony_ci srcRect.fBottom; 140cb93a386Sopenharmony_ci } 141cb93a386Sopenharmony_ci if (copiedDstRect.isEmpty() || 142cb93a386Sopenharmony_ci !copiedDstRect.intersect(SkIRect::MakeWH(kW, kH))) { 143cb93a386Sopenharmony_ci expectedResult = false; 144cb93a386Sopenharmony_ci } 145cb93a386Sopenharmony_ci if (sOrigin != dOrigin && dRenderable == GrRenderable::kNo) { 146cb93a386Sopenharmony_ci expectedResult = false; 147cb93a386Sopenharmony_ci } 148cb93a386Sopenharmony_ci 149cb93a386Sopenharmony_ci // To make the copied src rect correct we would apply any dst 150cb93a386Sopenharmony_ci // clipping back to the src rect, but we don't use it again so 151cb93a386Sopenharmony_ci // don't bother. 152cb93a386Sopenharmony_ci if (expectedResult != result) { 153cb93a386Sopenharmony_ci ERRORF(reporter, "Expected return value %d from copySurface, " 154cb93a386Sopenharmony_ci "got %d.", expectedResult, result); 155cb93a386Sopenharmony_ci continue; 156cb93a386Sopenharmony_ci } 157cb93a386Sopenharmony_ci 158cb93a386Sopenharmony_ci if (!expectedResult || !result) { 159cb93a386Sopenharmony_ci continue; 160cb93a386Sopenharmony_ci } 161cb93a386Sopenharmony_ci 162cb93a386Sopenharmony_ci sk_memset32(read.get(), 0, kW * kH); 163cb93a386Sopenharmony_ci GrPixmap readPM(ii, read.get(), kRowBytes); 164cb93a386Sopenharmony_ci if (!dstContext->readPixels(dContext, readPM, {0, 0})) { 165cb93a386Sopenharmony_ci ERRORF(reporter, "Error calling readPixels"); 166cb93a386Sopenharmony_ci continue; 167cb93a386Sopenharmony_ci } 168cb93a386Sopenharmony_ci 169cb93a386Sopenharmony_ci bool abort = false; 170cb93a386Sopenharmony_ci // Validate that pixels inside copiedDstRect received the correct 171cb93a386Sopenharmony_ci // value from src and that those outside were not modified. 172cb93a386Sopenharmony_ci for (int y = 0; y < kH && !abort; ++y) { 173cb93a386Sopenharmony_ci for (int x = 0; x < kW; ++x) { 174cb93a386Sopenharmony_ci uint32_t r = read.get()[y * kW + x]; 175cb93a386Sopenharmony_ci if (copiedDstRect.contains(x, y)) { 176cb93a386Sopenharmony_ci int sx = x - dstOffset.fX; 177cb93a386Sopenharmony_ci int sy = y - dstOffset.fY; 178cb93a386Sopenharmony_ci uint32_t s = srcPixels.get()[sy * kW + sx]; 179cb93a386Sopenharmony_ci if (s != r) { 180cb93a386Sopenharmony_ci ERRORF(reporter, "Expected dst %d,%d to contain " 181cb93a386Sopenharmony_ci "0x%08x copied from src location %d,%d. Got " 182cb93a386Sopenharmony_ci "0x%08x", x, y, s, sx, sy, r); 183cb93a386Sopenharmony_ci abort = true; 184cb93a386Sopenharmony_ci break; 185cb93a386Sopenharmony_ci } 186cb93a386Sopenharmony_ci } else { 187cb93a386Sopenharmony_ci uint32_t d = dstPixels.get()[y * kW + x]; 188cb93a386Sopenharmony_ci if (d != r) { 189cb93a386Sopenharmony_ci ERRORF(reporter, "Expected dst %d,%d to be " 190cb93a386Sopenharmony_ci "unmodified (0x%08x). Got 0x%08x", 191cb93a386Sopenharmony_ci x, y, d, r); 192cb93a386Sopenharmony_ci abort = true; 193cb93a386Sopenharmony_ci break; 194cb93a386Sopenharmony_ci } 195cb93a386Sopenharmony_ci } 196cb93a386Sopenharmony_ci } 197cb93a386Sopenharmony_ci } 198cb93a386Sopenharmony_ci } 199cb93a386Sopenharmony_ci } 200cb93a386Sopenharmony_ci } 201cb93a386Sopenharmony_ci } 202cb93a386Sopenharmony_ci } 203cb93a386Sopenharmony_ci } 204cb93a386Sopenharmony_ci } 205cb93a386Sopenharmony_ci} 206