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