xref: /aosp_15_r20/external/skia/tests/TextureOpTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2020 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkAlphaType.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkBlendMode.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkMatrix.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSamplingOptions.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSize.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/GpuTypes.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/GrBackendSurface.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/GrDirectContext.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/GrRecordingContext.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "include/gpu/ganesh/GrTypes.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/gpu/ganesh/GrTypesPriv.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkArenaAlloc.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/SkBackingFit.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/Swizzle.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrAppliedClip.h"
26*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrCaps.h"
27*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrColorSpaceXform.h"
28*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrDirectContextPriv.h"
29*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrProcessorSet.h"
30*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrProxyProvider.h"
31*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrRecordingContextPriv.h"
32*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrSamplerState.h"
33*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrSurfaceProxy.h"
34*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrSurfaceProxyView.h"
35*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/geometry/GrQuad.h"
36*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/ops/GrOp.h"
37*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/ops/OpsTask.h"
38*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/ops/TextureOp.h"
39*c8dee2aaSAndroid Build Coastguard Worker #include "tests/CtsEnforcement.h"
40*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
41*c8dee2aaSAndroid Build Coastguard Worker 
42*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
43*c8dee2aaSAndroid Build Coastguard Worker 
44*c8dee2aaSAndroid Build Coastguard Worker struct GrContextOptions;
45*c8dee2aaSAndroid Build Coastguard Worker 
46*c8dee2aaSAndroid Build Coastguard Worker using namespace skgpu::ganesh;
47*c8dee2aaSAndroid Build Coastguard Worker 
48*c8dee2aaSAndroid Build Coastguard Worker class OpsTaskTestingAccess {
49*c8dee2aaSAndroid Build Coastguard Worker public:
50*c8dee2aaSAndroid Build Coastguard Worker     typedef skgpu::ganesh::OpsTask::OpChain OpChain;
51*c8dee2aaSAndroid Build Coastguard Worker };
52*c8dee2aaSAndroid Build Coastguard Worker 
check_chain(OpsTaskTestingAccess::OpChain * chain,SkRect firstRect,SkRect lastRect,int expectedNumOps)53*c8dee2aaSAndroid Build Coastguard Worker static void check_chain(OpsTaskTestingAccess::OpChain* chain, SkRect firstRect, SkRect lastRect,
54*c8dee2aaSAndroid Build Coastguard Worker                         int expectedNumOps) {
55*c8dee2aaSAndroid Build Coastguard Worker     int actualNumOps = 0;
56*c8dee2aaSAndroid Build Coastguard Worker     for (const auto& op : GrOp::ChainRange<>(chain->head())) {
57*c8dee2aaSAndroid Build Coastguard Worker         ++actualNumOps;
58*c8dee2aaSAndroid Build Coastguard Worker 
59*c8dee2aaSAndroid Build Coastguard Worker         if (actualNumOps == 1) {
60*c8dee2aaSAndroid Build Coastguard Worker             SkAssertResult(op.bounds() == firstRect.makeOutset(0.5f, 0.5f));
61*c8dee2aaSAndroid Build Coastguard Worker         } else if (actualNumOps == expectedNumOps) {
62*c8dee2aaSAndroid Build Coastguard Worker             SkAssertResult(op.bounds() == lastRect.makeOutset(0.5f, 0.5f));
63*c8dee2aaSAndroid Build Coastguard Worker         }
64*c8dee2aaSAndroid Build Coastguard Worker     }
65*c8dee2aaSAndroid Build Coastguard Worker 
66*c8dee2aaSAndroid Build Coastguard Worker     SkAssertResult(actualNumOps == expectedNumOps);
67*c8dee2aaSAndroid Build Coastguard Worker }
68*c8dee2aaSAndroid Build Coastguard Worker 
create_proxy(GrRecordingContext * rContext)69*c8dee2aaSAndroid Build Coastguard Worker static sk_sp<GrSurfaceProxy> create_proxy(GrRecordingContext* rContext) {
70*c8dee2aaSAndroid Build Coastguard Worker     const GrCaps* caps = rContext->priv().caps();
71*c8dee2aaSAndroid Build Coastguard Worker 
72*c8dee2aaSAndroid Build Coastguard Worker     static constexpr SkISize kDimensions = {16, 16};
73*c8dee2aaSAndroid Build Coastguard Worker 
74*c8dee2aaSAndroid Build Coastguard Worker     const GrBackendFormat format = caps->getDefaultBackendFormat(GrColorType::kRGBA_8888,
75*c8dee2aaSAndroid Build Coastguard Worker                                                                  GrRenderable::kYes);
76*c8dee2aaSAndroid Build Coastguard Worker     return rContext->priv().proxyProvider()->createProxy(format,
77*c8dee2aaSAndroid Build Coastguard Worker                                                          kDimensions,
78*c8dee2aaSAndroid Build Coastguard Worker                                                          GrRenderable::kYes,
79*c8dee2aaSAndroid Build Coastguard Worker                                                          1,
80*c8dee2aaSAndroid Build Coastguard Worker                                                          skgpu::Mipmapped::kNo,
81*c8dee2aaSAndroid Build Coastguard Worker                                                          SkBackingFit::kExact,
82*c8dee2aaSAndroid Build Coastguard Worker                                                          skgpu::Budgeted::kNo,
83*c8dee2aaSAndroid Build Coastguard Worker                                                          GrProtected::kNo,
84*c8dee2aaSAndroid Build Coastguard Worker                                                          /*label=*/"TextureOpTest",
85*c8dee2aaSAndroid Build Coastguard Worker                                                          GrInternalSurfaceFlags::kNone);
86*c8dee2aaSAndroid Build Coastguard Worker }
87*c8dee2aaSAndroid Build Coastguard Worker 
create_op(GrDirectContext * dContext,SkRect rect,const GrSurfaceProxyView & proxyView,bool isAA)88*c8dee2aaSAndroid Build Coastguard Worker static GrOp::Owner create_op(GrDirectContext* dContext, SkRect rect,
89*c8dee2aaSAndroid Build Coastguard Worker                              const GrSurfaceProxyView& proxyView, bool isAA) {
90*c8dee2aaSAndroid Build Coastguard Worker     DrawQuad quad;
91*c8dee2aaSAndroid Build Coastguard Worker 
92*c8dee2aaSAndroid Build Coastguard Worker     quad.fDevice = GrQuad::MakeFromRect(rect.makeOutset(0.5f, 0.5f),  SkMatrix::I());
93*c8dee2aaSAndroid Build Coastguard Worker     quad.fLocal = GrQuad(rect);
94*c8dee2aaSAndroid Build Coastguard Worker     quad.fEdgeFlags = isAA ? GrQuadAAFlags::kAll : GrQuadAAFlags::kNone;
95*c8dee2aaSAndroid Build Coastguard Worker 
96*c8dee2aaSAndroid Build Coastguard Worker     return TextureOp::Make(dContext,
97*c8dee2aaSAndroid Build Coastguard Worker                            proxyView,
98*c8dee2aaSAndroid Build Coastguard Worker                            kPremul_SkAlphaType,
99*c8dee2aaSAndroid Build Coastguard Worker                            nullptr,
100*c8dee2aaSAndroid Build Coastguard Worker                            GrSamplerState::Filter::kNearest,
101*c8dee2aaSAndroid Build Coastguard Worker                            GrSamplerState::MipmapMode::kNone,
102*c8dee2aaSAndroid Build Coastguard Worker                            {1.f, 1.f, 1.f, 1.f},
103*c8dee2aaSAndroid Build Coastguard Worker                            TextureOp::Saturate::kYes,
104*c8dee2aaSAndroid Build Coastguard Worker                            SkBlendMode::kSrcOver,
105*c8dee2aaSAndroid Build Coastguard Worker                            isAA ? GrAAType::kCoverage
106*c8dee2aaSAndroid Build Coastguard Worker                                 : GrAAType::kNone,
107*c8dee2aaSAndroid Build Coastguard Worker                            &quad,
108*c8dee2aaSAndroid Build Coastguard Worker                            nullptr);
109*c8dee2aaSAndroid Build Coastguard Worker }
110*c8dee2aaSAndroid Build Coastguard Worker 
111*c8dee2aaSAndroid Build Coastguard Worker // This unit test exercises the crbug.com/1112259 case.
DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(TextureOpTest,reporter,ctxInfo,CtsEnforcement::kNever)112*c8dee2aaSAndroid Build Coastguard Worker DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(TextureOpTest, reporter, ctxInfo, CtsEnforcement::kNever) {
113*c8dee2aaSAndroid Build Coastguard Worker     GrDirectContext* dContext = ctxInfo.directContext();
114*c8dee2aaSAndroid Build Coastguard Worker     const GrCaps* caps = dContext->priv().caps();
115*c8dee2aaSAndroid Build Coastguard Worker     SkArenaAlloc arena{nullptr, 0, 1024};
116*c8dee2aaSAndroid Build Coastguard Worker     auto auditTrail = dContext->priv().auditTrail();
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker     if (!caps->dynamicStateArrayGeometryProcessorTextureSupport()) {
119*c8dee2aaSAndroid Build Coastguard Worker         // This test requires chaining
120*c8dee2aaSAndroid Build Coastguard Worker         return;
121*c8dee2aaSAndroid Build Coastguard Worker     }
122*c8dee2aaSAndroid Build Coastguard Worker 
123*c8dee2aaSAndroid Build Coastguard Worker     GrSurfaceProxyView proxyViewA(create_proxy(dContext),
124*c8dee2aaSAndroid Build Coastguard Worker                                   kTopLeft_GrSurfaceOrigin,
125*c8dee2aaSAndroid Build Coastguard Worker                                   skgpu::Swizzle::RGBA());
126*c8dee2aaSAndroid Build Coastguard Worker     GrSurfaceProxyView proxyViewB(create_proxy(dContext),
127*c8dee2aaSAndroid Build Coastguard Worker                                   kTopLeft_GrSurfaceOrigin,
128*c8dee2aaSAndroid Build Coastguard Worker                                   skgpu::Swizzle::RGBA());
129*c8dee2aaSAndroid Build Coastguard Worker     GrSurfaceProxyView proxyViewC(create_proxy(dContext),
130*c8dee2aaSAndroid Build Coastguard Worker                                   kTopLeft_GrSurfaceOrigin,
131*c8dee2aaSAndroid Build Coastguard Worker                                   skgpu::Swizzle::RGBA());
132*c8dee2aaSAndroid Build Coastguard Worker 
133*c8dee2aaSAndroid Build Coastguard Worker     static const SkRect kOpARect{  0,  0, 16, 16 };
134*c8dee2aaSAndroid Build Coastguard Worker     static const SkRect kOpBRect{ 32,  0, 48, 16 };
135*c8dee2aaSAndroid Build Coastguard Worker     static const SkRect kOpCRect{  0, 32, 16, 48 };
136*c8dee2aaSAndroid Build Coastguard Worker     static const SkRect kOpDRect{ 32, 32, 48, 48 };
137*c8dee2aaSAndroid Build Coastguard Worker 
138*c8dee2aaSAndroid Build Coastguard Worker     // opA & opB can chain together but can't merge bc they have different proxies
139*c8dee2aaSAndroid Build Coastguard Worker     GrOp::Owner opA = create_op(dContext, kOpARect, proxyViewA, false);
140*c8dee2aaSAndroid Build Coastguard Worker     GrOp::Owner opB = create_op(dContext, kOpBRect, proxyViewB, false);
141*c8dee2aaSAndroid Build Coastguard Worker 
142*c8dee2aaSAndroid Build Coastguard Worker     GrAppliedClip noClip = GrAppliedClip::Disabled();
143*c8dee2aaSAndroid Build Coastguard Worker     OpsTaskTestingAccess::OpChain chain1(std::move(opA), GrProcessorSet::EmptySetAnalysis(),
144*c8dee2aaSAndroid Build Coastguard Worker                                          &noClip, nullptr);
145*c8dee2aaSAndroid Build Coastguard Worker     chain1.appendOp(std::move(opB), GrProcessorSet::EmptySetAnalysis(), nullptr, &noClip, *caps,
146*c8dee2aaSAndroid Build Coastguard Worker                     &arena, dContext->priv().auditTrail());
147*c8dee2aaSAndroid Build Coastguard Worker     check_chain(&chain1, kOpARect, kOpBRect, 2);
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker     // opC & opD can also chain together but can't merge (bc, again, they have different
150*c8dee2aaSAndroid Build Coastguard Worker     // proxies). Note, however, that opA and opD do share a proxy so can be merged if opA's
151*c8dee2aaSAndroid Build Coastguard Worker     // anti-aliasing is upgraded to coverage.
152*c8dee2aaSAndroid Build Coastguard Worker     GrOp::Owner opC = create_op(dContext, kOpCRect, proxyViewC, true);
153*c8dee2aaSAndroid Build Coastguard Worker     GrOp::Owner opD = create_op(dContext, kOpDRect, proxyViewA, true);
154*c8dee2aaSAndroid Build Coastguard Worker 
155*c8dee2aaSAndroid Build Coastguard Worker     OpsTaskTestingAccess::OpChain chain2(std::move(opC), GrProcessorSet::EmptySetAnalysis(),
156*c8dee2aaSAndroid Build Coastguard Worker                                          &noClip, nullptr);
157*c8dee2aaSAndroid Build Coastguard Worker     chain2.appendOp(std::move(opD), GrProcessorSet::EmptySetAnalysis(), nullptr, &noClip, *caps,
158*c8dee2aaSAndroid Build Coastguard Worker                     &arena, auditTrail);
159*c8dee2aaSAndroid Build Coastguard Worker     check_chain(&chain2, kOpCRect, kOpDRect, 2);
160*c8dee2aaSAndroid Build Coastguard Worker 
161*c8dee2aaSAndroid Build Coastguard Worker     // opA and opD, now in separate chains, can merge when the two chains are combined while
162*c8dee2aaSAndroid Build Coastguard Worker     // opB and opC can still only chain.
163*c8dee2aaSAndroid Build Coastguard Worker     chain1.prependChain(&chain2, *caps, &arena, auditTrail);
164*c8dee2aaSAndroid Build Coastguard Worker 
165*c8dee2aaSAndroid Build Coastguard Worker     // We should end up with the chain
166*c8dee2aaSAndroid Build Coastguard Worker     //   opC - opD/opA - opB
167*c8dee2aaSAndroid Build Coastguard Worker     check_chain(&chain1, kOpCRect, kOpBRect, 3);
168*c8dee2aaSAndroid Build Coastguard Worker 
169*c8dee2aaSAndroid Build Coastguard Worker     chain1.deleteOps();
170*c8dee2aaSAndroid Build Coastguard Worker }
171