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